Commit 8d18fba2 authored by Henrik Rydberg's avatar Henrik Rydberg

Input: Break out MT data

Move all MT-related things to a separate place. This saves some
bytes for non-mt input devices, and prepares for new MT features.
Reviewed-and-tested-by: default avatarBenjamin Tissoires <benjamin.tissoires@enac.fr>
Tested-by: default avatarPing Cheng <pingc@wacom.com>
Acked-by: default avatarDmitry Torokhov <dmitry.torokhov@gmail.com>
Signed-off-by: default avatarHenrik Rydberg <rydberg@euromail.se>
parent 4cbe5a55
...@@ -653,20 +653,22 @@ static int evdev_handle_mt_request(struct input_dev *dev, ...@@ -653,20 +653,22 @@ static int evdev_handle_mt_request(struct input_dev *dev,
unsigned int size, unsigned int size,
int __user *ip) int __user *ip)
{ {
const struct input_mt_slot *mt = dev->mt; const struct input_mt *mt = dev->mt;
unsigned int code; unsigned int code;
int max_slots; int max_slots;
int i; int i;
if (get_user(code, &ip[0])) if (get_user(code, &ip[0]))
return -EFAULT; return -EFAULT;
if (!input_is_mt_value(code)) if (!mt || !input_is_mt_value(code))
return -EINVAL; return -EINVAL;
max_slots = (size - sizeof(__u32)) / sizeof(__s32); max_slots = (size - sizeof(__u32)) / sizeof(__s32);
for (i = 0; i < dev->mtsize && i < max_slots; i++) for (i = 0; i < mt->num_slots && i < max_slots; i++) {
if (put_user(input_mt_get_value(&mt[i], code), &ip[1 + i])) int value = input_mt_get_value(&mt->slots[i], code);
if (put_user(value, &ip[1 + i]))
return -EFAULT; return -EFAULT;
}
return 0; return 0;
} }
......
...@@ -27,26 +27,28 @@ ...@@ -27,26 +27,28 @@
*/ */
int input_mt_init_slots(struct input_dev *dev, unsigned int num_slots) int input_mt_init_slots(struct input_dev *dev, unsigned int num_slots)
{ {
struct input_mt *mt = dev->mt;
int i; int i;
if (!num_slots) if (!num_slots)
return 0; return 0;
if (dev->mt) if (mt)
return dev->mtsize != num_slots ? -EINVAL : 0; return mt->num_slots != num_slots ? -EINVAL : 0;
dev->mt = kcalloc(num_slots, sizeof(struct input_mt_slot), GFP_KERNEL); mt = kzalloc(sizeof(*mt) + num_slots * sizeof(*mt->slots), GFP_KERNEL);
if (!dev->mt) if (!mt)
return -ENOMEM; return -ENOMEM;
dev->mtsize = num_slots; mt->num_slots = num_slots;
input_set_abs_params(dev, ABS_MT_SLOT, 0, num_slots - 1, 0, 0); input_set_abs_params(dev, ABS_MT_SLOT, 0, num_slots - 1, 0, 0);
input_set_abs_params(dev, ABS_MT_TRACKING_ID, 0, TRKID_MAX, 0, 0); input_set_abs_params(dev, ABS_MT_TRACKING_ID, 0, TRKID_MAX, 0, 0);
input_set_events_per_packet(dev, 6 * num_slots); input_set_events_per_packet(dev, 6 * num_slots);
/* Mark slots as 'unused' */ /* Mark slots as 'unused' */
for (i = 0; i < num_slots; i++) for (i = 0; i < num_slots; i++)
input_mt_set_value(&dev->mt[i], ABS_MT_TRACKING_ID, -1); input_mt_set_value(&mt->slots[i], ABS_MT_TRACKING_ID, -1);
dev->mt = mt;
return 0; return 0;
} }
EXPORT_SYMBOL(input_mt_init_slots); EXPORT_SYMBOL(input_mt_init_slots);
...@@ -62,9 +64,6 @@ void input_mt_destroy_slots(struct input_dev *dev) ...@@ -62,9 +64,6 @@ void input_mt_destroy_slots(struct input_dev *dev)
{ {
kfree(dev->mt); kfree(dev->mt);
dev->mt = NULL; dev->mt = NULL;
dev->mtsize = 0;
dev->slot = 0;
dev->trkid = 0;
} }
EXPORT_SYMBOL(input_mt_destroy_slots); EXPORT_SYMBOL(input_mt_destroy_slots);
...@@ -83,18 +82,19 @@ EXPORT_SYMBOL(input_mt_destroy_slots); ...@@ -83,18 +82,19 @@ EXPORT_SYMBOL(input_mt_destroy_slots);
void input_mt_report_slot_state(struct input_dev *dev, void input_mt_report_slot_state(struct input_dev *dev,
unsigned int tool_type, bool active) unsigned int tool_type, bool active)
{ {
struct input_mt_slot *mt; struct input_mt *mt = dev->mt;
struct input_mt_slot *slot;
int id; int id;
if (!dev->mt || !active) { if (!mt || !active) {
input_event(dev, EV_ABS, ABS_MT_TRACKING_ID, -1); input_event(dev, EV_ABS, ABS_MT_TRACKING_ID, -1);
return; return;
} }
mt = &dev->mt[dev->slot]; slot = &mt->slots[mt->slot];
id = input_mt_get_value(mt, ABS_MT_TRACKING_ID); id = input_mt_get_value(slot, ABS_MT_TRACKING_ID);
if (id < 0 || input_mt_get_value(mt, ABS_MT_TOOL_TYPE) != tool_type) if (id < 0 || input_mt_get_value(slot, ABS_MT_TOOL_TYPE) != tool_type)
id = input_mt_new_trkid(dev); id = input_mt_new_trkid(mt);
input_event(dev, EV_ABS, ABS_MT_TRACKING_ID, id); input_event(dev, EV_ABS, ABS_MT_TRACKING_ID, id);
input_event(dev, EV_ABS, ABS_MT_TOOL_TYPE, tool_type); input_event(dev, EV_ABS, ABS_MT_TOOL_TYPE, tool_type);
...@@ -135,13 +135,19 @@ EXPORT_SYMBOL(input_mt_report_finger_count); ...@@ -135,13 +135,19 @@ EXPORT_SYMBOL(input_mt_report_finger_count);
*/ */
void input_mt_report_pointer_emulation(struct input_dev *dev, bool use_count) void input_mt_report_pointer_emulation(struct input_dev *dev, bool use_count)
{ {
struct input_mt_slot *oldest = NULL; struct input_mt *mt = dev->mt;
int oldid = dev->trkid; struct input_mt_slot *oldest;
int count = 0; int oldid, count, i;
int i;
if (!mt)
return;
oldest = 0;
oldid = mt->trkid;
count = 0;
for (i = 0; i < dev->mtsize; ++i) { for (i = 0; i < mt->num_slots; ++i) {
struct input_mt_slot *ps = &dev->mt[i]; struct input_mt_slot *ps = &mt->slots[i];
int id = input_mt_get_value(ps, ABS_MT_TRACKING_ID); int id = input_mt_get_value(ps, ABS_MT_TRACKING_ID);
if (id < 0) if (id < 0)
......
...@@ -166,6 +166,7 @@ static void input_stop_autorepeat(struct input_dev *dev) ...@@ -166,6 +166,7 @@ static void input_stop_autorepeat(struct input_dev *dev)
static int input_handle_abs_event(struct input_dev *dev, static int input_handle_abs_event(struct input_dev *dev,
unsigned int code, int *pval) unsigned int code, int *pval)
{ {
struct input_mt *mt = dev->mt;
bool is_mt_event; bool is_mt_event;
int *pold; int *pold;
...@@ -174,8 +175,8 @@ static int input_handle_abs_event(struct input_dev *dev, ...@@ -174,8 +175,8 @@ static int input_handle_abs_event(struct input_dev *dev,
* "Stage" the event; we'll flush it later, when we * "Stage" the event; we'll flush it later, when we
* get actual touch data. * get actual touch data.
*/ */
if (*pval >= 0 && *pval < dev->mtsize) if (mt && *pval >= 0 && *pval < mt->num_slots)
dev->slot = *pval; mt->slot = *pval;
return INPUT_IGNORE_EVENT; return INPUT_IGNORE_EVENT;
} }
...@@ -184,9 +185,8 @@ static int input_handle_abs_event(struct input_dev *dev, ...@@ -184,9 +185,8 @@ static int input_handle_abs_event(struct input_dev *dev,
if (!is_mt_event) { if (!is_mt_event) {
pold = &dev->absinfo[code].value; pold = &dev->absinfo[code].value;
} else if (dev->mt) { } else if (mt) {
struct input_mt_slot *mtslot = &dev->mt[dev->slot]; pold = &mt->slots[mt->slot].abs[code - ABS_MT_FIRST];
pold = &mtslot->abs[code - ABS_MT_FIRST];
} else { } else {
/* /*
* Bypass filtering for multi-touch events when * Bypass filtering for multi-touch events when
...@@ -205,9 +205,9 @@ static int input_handle_abs_event(struct input_dev *dev, ...@@ -205,9 +205,9 @@ static int input_handle_abs_event(struct input_dev *dev,
} }
/* Flush pending "slot" event */ /* Flush pending "slot" event */
if (is_mt_event && dev->slot != input_abs_get_val(dev, ABS_MT_SLOT)) { if (is_mt_event && mt && mt->slot != input_abs_get_val(dev, ABS_MT_SLOT)) {
input_abs_set_val(dev, ABS_MT_SLOT, dev->slot); input_abs_set_val(dev, ABS_MT_SLOT, mt->slot);
input_pass_event(dev, EV_ABS, ABS_MT_SLOT, dev->slot); input_pass_event(dev, EV_ABS, ABS_MT_SLOT, mt->slot);
} }
return INPUT_PASS_TO_HANDLERS; return INPUT_PASS_TO_HANDLERS;
...@@ -1751,8 +1751,8 @@ static unsigned int input_estimate_events_per_packet(struct input_dev *dev) ...@@ -1751,8 +1751,8 @@ static unsigned int input_estimate_events_per_packet(struct input_dev *dev)
int i; int i;
unsigned int events; unsigned int events;
if (dev->mtsize) { if (dev->mt) {
mt_slots = dev->mtsize; mt_slots = dev->mt->num_slots;
} else if (test_bit(ABS_MT_TRACKING_ID, dev->absbit)) { } else if (test_bit(ABS_MT_TRACKING_ID, dev->absbit)) {
mt_slots = dev->absinfo[ABS_MT_TRACKING_ID].maximum - mt_slots = dev->absinfo[ABS_MT_TRACKING_ID].maximum -
dev->absinfo[ABS_MT_TRACKING_ID].minimum + 1, dev->absinfo[ABS_MT_TRACKING_ID].minimum + 1,
......
...@@ -1203,11 +1203,7 @@ struct ff_effect { ...@@ -1203,11 +1203,7 @@ struct ff_effect {
* software autorepeat * software autorepeat
* @timer: timer for software autorepeat * @timer: timer for software autorepeat
* @rep: current values for autorepeat parameters (delay, rate) * @rep: current values for autorepeat parameters (delay, rate)
* @mt: pointer to array of struct input_mt_slot holding current values * @mt: pointer to multitouch state
* of tracked contacts
* @mtsize: number of MT slots the device uses
* @slot: MT slot currently being transmitted
* @trkid: stores MT tracking ID for the current contact
* @absinfo: array of &struct input_absinfo elements holding information * @absinfo: array of &struct input_absinfo elements holding information
* about absolute axes (current value, min, max, flat, fuzz, * about absolute axes (current value, min, max, flat, fuzz,
* resolution) * resolution)
...@@ -1287,10 +1283,7 @@ struct input_dev { ...@@ -1287,10 +1283,7 @@ struct input_dev {
int rep[REP_CNT]; int rep[REP_CNT];
struct input_mt_slot *mt; struct input_mt *mt;
int mtsize;
int slot;
int trkid;
struct input_absinfo *absinfo; struct input_absinfo *absinfo;
......
...@@ -23,6 +23,20 @@ struct input_mt_slot { ...@@ -23,6 +23,20 @@ struct input_mt_slot {
int abs[ABS_MT_LAST - ABS_MT_FIRST + 1]; int abs[ABS_MT_LAST - ABS_MT_FIRST + 1];
}; };
/**
* struct input_mt - state of tracked contacts
* @trkid: stores MT tracking ID for the next contact
* @num_slots: number of MT slots the device uses
* @slot: MT slot currently being transmitted
* @slots: array of slots holding current values of tracked contacts
*/
struct input_mt {
int trkid;
int num_slots;
int slot;
struct input_mt_slot slots[];
};
static inline void input_mt_set_value(struct input_mt_slot *slot, static inline void input_mt_set_value(struct input_mt_slot *slot,
unsigned code, int value) unsigned code, int value)
{ {
...@@ -38,9 +52,9 @@ static inline int input_mt_get_value(const struct input_mt_slot *slot, ...@@ -38,9 +52,9 @@ static inline int input_mt_get_value(const struct input_mt_slot *slot,
int input_mt_init_slots(struct input_dev *dev, unsigned int num_slots); int input_mt_init_slots(struct input_dev *dev, unsigned int num_slots);
void input_mt_destroy_slots(struct input_dev *dev); void input_mt_destroy_slots(struct input_dev *dev);
static inline int input_mt_new_trkid(struct input_dev *dev) static inline int input_mt_new_trkid(struct input_mt *mt)
{ {
return dev->trkid++ & TRKID_MAX; return mt->trkid++ & TRKID_MAX;
} }
static inline void input_mt_slot(struct input_dev *dev, int slot) static inline void input_mt_slot(struct input_dev *dev, int slot)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment