1 /*- 2 * Copyright (c) 2016 Vladimir Kondratyev <wulf@cicgroup.ru> 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24 * SUCH DAMAGE. 25 * 26 * $FreeBSD$ 27 */ 28 29 #include <sys/param.h> 30 #include <sys/malloc.h> 31 #include <sys/lock.h> 32 #include <sys/mutex.h> 33 #include <sys/systm.h> 34 35 #include <dev/evdev/input.h> 36 #include <dev/evdev/evdev.h> 37 #include <dev/evdev/evdev_private.h> 38 39 #ifdef DEBUG 40 #define debugf(fmt, args...) printf("evdev: " fmt "\n", ##args) 41 #else 42 #define debugf(fmt, args...) 43 #endif 44 45 static uint16_t evdev_fngmap[] = { 46 BTN_TOOL_FINGER, 47 BTN_TOOL_DOUBLETAP, 48 BTN_TOOL_TRIPLETAP, 49 BTN_TOOL_QUADTAP, 50 BTN_TOOL_QUINTTAP, 51 }; 52 53 static uint16_t evdev_mtstmap[][2] = { 54 { ABS_MT_POSITION_X, ABS_X }, 55 { ABS_MT_POSITION_Y, ABS_Y }, 56 { ABS_MT_PRESSURE, ABS_PRESSURE }, 57 { ABS_MT_TOUCH_MAJOR, ABS_TOOL_WIDTH }, 58 }; 59 60 struct evdev_mt_slot { 61 uint64_t ev_report; 62 int32_t ev_mt_states[MT_CNT]; 63 }; 64 65 struct evdev_mt { 66 int32_t ev_mt_last_reported_slot; 67 struct evdev_mt_slot ev_mt_slots[]; 68 }; 69 70 void 71 evdev_mt_init(struct evdev_dev *evdev) 72 { 73 int32_t slot, slots; 74 75 slots = MAXIMAL_MT_SLOT(evdev) + 1; 76 77 evdev->ev_mt = malloc(offsetof(struct evdev_mt, ev_mt_slots) + 78 sizeof(struct evdev_mt_slot) * slots, M_EVDEV, M_WAITOK | M_ZERO); 79 80 /* Initialize multitouch protocol type B states */ 81 for (slot = 0; slot < slots; slot++) { 82 /* 83 * .ev_report should not be initialized to initial value of 84 * report counter (0) as it brokes free slot detection in 85 * evdev_get_mt_slot_by_tracking_id. So initialize it to -1 86 */ 87 evdev->ev_mt->ev_mt_slots[slot] = (struct evdev_mt_slot) { 88 .ev_report = 0xFFFFFFFFFFFFFFFFULL, 89 .ev_mt_states[ABS_MT_INDEX(ABS_MT_TRACKING_ID)] = -1, 90 }; 91 } 92 93 if (bit_test(evdev->ev_flags, EVDEV_FLAG_MT_STCOMPAT)) 94 evdev_support_mt_compat(evdev); 95 } 96 97 void 98 evdev_mt_free(struct evdev_dev *evdev) 99 { 100 101 free(evdev->ev_mt, M_EVDEV); 102 } 103 104 int32_t 105 evdev_get_last_mt_slot(struct evdev_dev *evdev) 106 { 107 108 return (evdev->ev_mt->ev_mt_last_reported_slot); 109 } 110 111 void 112 evdev_set_last_mt_slot(struct evdev_dev *evdev, int32_t slot) 113 { 114 115 evdev->ev_mt->ev_mt_slots[slot].ev_report = evdev->ev_report_count; 116 evdev->ev_mt->ev_mt_last_reported_slot = slot; 117 } 118 119 inline int32_t 120 evdev_get_mt_value(struct evdev_dev *evdev, int32_t slot, int16_t code) 121 { 122 123 return (evdev->ev_mt-> 124 ev_mt_slots[slot].ev_mt_states[ABS_MT_INDEX(code)]); 125 } 126 127 inline void 128 evdev_set_mt_value(struct evdev_dev *evdev, int32_t slot, int16_t code, 129 int32_t value) 130 { 131 132 evdev->ev_mt->ev_mt_slots[slot].ev_mt_states[ABS_MT_INDEX(code)] = 133 value; 134 } 135 136 int32_t 137 evdev_get_mt_slot_by_tracking_id(struct evdev_dev *evdev, int32_t tracking_id) 138 { 139 int32_t tr_id, slot, free_slot = -1; 140 141 for (slot = 0; slot <= MAXIMAL_MT_SLOT(evdev); slot++) { 142 tr_id = evdev_get_mt_value(evdev, slot, ABS_MT_TRACKING_ID); 143 if (tr_id == tracking_id) 144 return (slot); 145 /* 146 * Its possible that slot will be reassigned in a place of just 147 * released one within the same report. To avoid this compare 148 * report counter with slot`s report number updated with each 149 * ABS_MT_TRACKING_ID change. 150 */ 151 if (free_slot == -1 && tr_id == -1 && 152 evdev->ev_mt->ev_mt_slots[slot].ev_report != 153 evdev->ev_report_count) 154 free_slot = slot; 155 } 156 157 return (free_slot); 158 } 159 160 void 161 evdev_support_nfingers(struct evdev_dev *evdev, int32_t nfingers) 162 { 163 int32_t i; 164 165 for (i = 0; i < MIN(nitems(evdev_fngmap), nfingers); i++) 166 evdev_support_key(evdev, evdev_fngmap[i]); 167 } 168 169 void 170 evdev_support_mt_compat(struct evdev_dev *evdev) 171 { 172 int32_t i; 173 174 if (evdev->ev_absinfo == NULL) 175 return; 176 177 evdev_support_event(evdev, EV_KEY); 178 evdev_support_key(evdev, BTN_TOUCH); 179 180 /* Touchscreens should not advertise tap tool capabilities */ 181 if (!bit_test(evdev->ev_prop_flags, INPUT_PROP_DIRECT)) 182 evdev_support_nfingers(evdev, MAXIMAL_MT_SLOT(evdev) + 1); 183 184 /* Echo 0-th MT-slot as ST-slot */ 185 for (i = 0; i < nitems(evdev_mtstmap); i++) 186 if (bit_test(evdev->ev_abs_flags, evdev_mtstmap[i][0])) 187 evdev_support_abs(evdev, evdev_mtstmap[i][1], 188 evdev->ev_absinfo[evdev_mtstmap[i][0]].value, 189 evdev->ev_absinfo[evdev_mtstmap[i][0]].minimum, 190 evdev->ev_absinfo[evdev_mtstmap[i][0]].maximum, 191 evdev->ev_absinfo[evdev_mtstmap[i][0]].fuzz, 192 evdev->ev_absinfo[evdev_mtstmap[i][0]].flat, 193 evdev->ev_absinfo[evdev_mtstmap[i][0]].resolution); 194 } 195 196 static int32_t 197 evdev_count_fingers(struct evdev_dev *evdev) 198 { 199 int32_t nfingers = 0, i; 200 201 for (i = 0; i <= MAXIMAL_MT_SLOT(evdev); i++) 202 if (evdev_get_mt_value(evdev, i, ABS_MT_TRACKING_ID) != -1) 203 nfingers++; 204 205 return (nfingers); 206 } 207 208 static void 209 evdev_send_nfingers(struct evdev_dev *evdev, int32_t nfingers) 210 { 211 int32_t i; 212 213 EVDEV_LOCK_ASSERT(evdev); 214 215 if (nfingers > nitems(evdev_fngmap)) 216 nfingers = nitems(evdev_fngmap); 217 218 for (i = 0; i < nitems(evdev_fngmap); i++) 219 evdev_send_event(evdev, EV_KEY, evdev_fngmap[i], 220 nfingers == i + 1); 221 } 222 223 void 224 evdev_push_nfingers(struct evdev_dev *evdev, int32_t nfingers) 225 { 226 227 if (evdev->ev_lock_type == EV_LOCK_INTERNAL) 228 EVDEV_LOCK(evdev); 229 else 230 EVDEV_LOCK_ASSERT(evdev); 231 evdev_send_nfingers(evdev, nfingers); 232 if (evdev->ev_lock_type == EV_LOCK_INTERNAL) 233 EVDEV_UNLOCK(evdev); 234 } 235 236 void 237 evdev_send_mt_compat(struct evdev_dev *evdev) 238 { 239 int32_t nfingers, i; 240 241 EVDEV_LOCK_ASSERT(evdev); 242 243 nfingers = evdev_count_fingers(evdev); 244 evdev_send_event(evdev, EV_KEY, BTN_TOUCH, nfingers > 0); 245 246 if (evdev_get_mt_value(evdev, 0, ABS_MT_TRACKING_ID) != -1) 247 /* Echo 0-th MT-slot as ST-slot */ 248 for (i = 0; i < nitems(evdev_mtstmap); i++) 249 if (bit_test(evdev->ev_abs_flags, evdev_mtstmap[i][1])) 250 evdev_send_event(evdev, EV_ABS, 251 evdev_mtstmap[i][1], 252 evdev_get_mt_value(evdev, 0, 253 evdev_mtstmap[i][0])); 254 255 /* Touchscreens should not report tool taps */ 256 if (!bit_test(evdev->ev_prop_flags, INPUT_PROP_DIRECT)) 257 evdev_send_nfingers(evdev, nfingers); 258 259 if (nfingers == 0) 260 evdev_send_event(evdev, EV_ABS, ABS_PRESSURE, 0); 261 } 262 263 void 264 evdev_push_mt_compat(struct evdev_dev *evdev) 265 { 266 267 if (evdev->ev_lock_type == EV_LOCK_INTERNAL) 268 EVDEV_LOCK(evdev); 269 else 270 EVDEV_LOCK_ASSERT(evdev); 271 evdev_send_mt_compat(evdev); 272 if (evdev->ev_lock_type == EV_LOCK_INTERNAL) 273 EVDEV_UNLOCK(evdev); 274 } 275 276 void 277 evdev_send_mt_autorel(struct evdev_dev *evdev) 278 { 279 int32_t slot; 280 281 EVDEV_LOCK_ASSERT(evdev); 282 283 for (slot = 0; slot <= MAXIMAL_MT_SLOT(evdev); slot++) { 284 if (evdev->ev_mt->ev_mt_slots[slot].ev_report != 285 evdev->ev_report_count && 286 evdev_get_mt_value(evdev, slot, ABS_MT_TRACKING_ID) != -1){ 287 evdev_send_event(evdev, EV_ABS, ABS_MT_SLOT, slot); 288 evdev_send_event(evdev, EV_ABS, ABS_MT_TRACKING_ID, 289 -1); 290 } 291 } 292 } 293