Recompute keybit when keycode table changes.
Stricter checks on input keycode/scancode values.
int getkeycode(unsigned int scancode)
{
struct input_handle *handle;
- unsigned int keycode;
+ struct input_dev *dev = NULL;
for (handle = kbd_handler.handle; handle; handle = handle->hnext)
- if (handle->dev->keycodesize) break;
+ if (handle->dev->keycodesize) { dev = handle->dev; break; }
- if (!handle->dev->keycodesize)
+ if (!dev)
return -ENODEV;
- switch (handle->dev->keycodesize) {
- case 1: keycode = *(u8*)(handle->dev->keycode + scancode); break;
- case 2: keycode = *(u16*)(handle->dev->keycode + scancode * 2); break;
- case 4: keycode = *(u32*)(handle->dev->keycode + scancode * 4); break;
- default: return -EINVAL;
- }
+ if (scancode < 0 || scancode >= dev->keycodemax)
+ return -EINVAL;
- return keycode;
+ return INPUT_KEYCODE(dev, scancode);
}
int setkeycode(unsigned int scancode, unsigned int keycode)
{
struct input_handle *handle;
+ struct input_dev *dev = NULL;
+ int i, oldkey;
for (handle = kbd_handler.handle; handle; handle = handle->hnext)
- if (handle->dev->keycodesize) break;
+ if (handle->dev->keycodesize) { dev = handle->dev; break; }
- if (!handle->dev->keycodesize)
+ if (!dev)
return -ENODEV;
- switch (handle->dev->keycodesize) {
- case 1: *(u8*)(handle->dev->keycode + scancode) = keycode; break;
- case 2: *(u16*)(handle->dev->keycode + scancode * 2) = keycode; break;
- case 4: *(u32*)(handle->dev->keycode + scancode * 4) = keycode; break;
- }
+ if (scancode < 0 || scancode >= dev->keycodemax)
+ return -EINVAL;
+
+ oldkey = INPUT_KEYCODE(dev, scancode);
+ INPUT_KEYCODE(dev, scancode) = keycode;
+
+ for (i = 0; i < dev->keycodemax; i++)
+ if(INPUT_KEYCODE(dev, scancode) == oldkey)
+ break;
+ if (i == dev->keycodemax)
+ clear_bit(oldkey, dev->keybit);
+ set_bit(keycode, dev->keybit);
return 0;
}
struct evdev *evdev = list->evdev;
struct input_dev *dev = evdev->handle.dev;
struct input_absinfo abs;
- int t, u;
+ int i, t, u;
if (!evdev->exist) return -ENODEV;
case EVIOCGKEYCODE:
if (get_user(t, ((int *) arg) + 0)) return -EFAULT;
- if (t < 0 || t > dev->keycodemax) return -EINVAL;
- switch (dev->keycodesize) {
- case 1: u = *(u8*)(dev->keycode + t); break;
- case 2: u = *(u16*)(dev->keycode + t * 2); break;
- case 4: u = *(u32*)(dev->keycode + t * 4); break;
- default: return -EINVAL;
- }
- if (put_user(u, ((int *) arg) + 1)) return -EFAULT;
+ if (t < 0 || t > dev->keycodemax || !dev->keycodesize) return -EINVAL;
+ if (put_user(INPUT_KEYCODE(dev, t), ((int *) arg) + 1)) return -EFAULT;
return 0;
case EVIOCSKEYCODE:
if (get_user(t, ((int *) arg) + 0)) return -EFAULT;
- if (t < 0 || t > dev->keycodemax) return -EINVAL;
- if (get_user(u, ((int *) arg) + 1)) return -EFAULT;
- switch (dev->keycodesize) {
- case 1: *(u8*)(dev->keycode + t) = u; break;
- case 2: *(u16*)(dev->keycode + t * 2) = u; break;
- case 4: *(u32*)(dev->keycode + t * 4) = u; break;
- default: return -EINVAL;
- }
+ if (t < 0 || t > dev->keycodemax || !dev->keycodesize) return -EINVAL;
+ u = INPUT_KEYCODE(dev, t);
+ if (get_user(INPUT_KEYCODE(dev, t), ((int *) arg) + 1)) return -EFAULT;
+
+ for (i = 0; i < dev->keycodemax; i++)
+ if(INPUT_KEYCODE(dev, t) == u) break;
+ if (i == dev->keycodemax) clear_bit(u, dev->keybit);
+ set_bit(INPUT_KEYCODE(dev, t), dev->keybit);
+
return 0;
case EVIOCSFF:
#define BIT(x) (1UL<<((x)%BITS_PER_LONG))
#define LONG(x) ((x)/BITS_PER_LONG)
+#define INPUT_KEYCODE(dev, scancode) ((dev->keycodesize == 1) ? ((u8*)dev->keycode)[scancode] : \
+ ((dev->keycodesize == 1) ? ((u16*)dev->keycode)[scancode] : (((u32*)dev->keycode)[scancode])))
+
struct input_dev {
void *private;