trackpoint.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. /*
  2. * Stephen Evanchik <evanchsa@gmail.com>
  3. *
  4. * This program is free software; you can redistribute it and/or modify it
  5. * under the terms of the GNU General Public License version 2 as published by
  6. * the Free Software Foundation.
  7. *
  8. * Trademarks are the property of their respective owners.
  9. */
  10. #include <linux/slab.h>
  11. #include <linux/delay.h>
  12. #include <linux/serio.h>
  13. #include <linux/module.h>
  14. #include <linux/input.h>
  15. #include <linux/libps2.h>
  16. #include <linux/proc_fs.h>
  17. #include <linux/uaccess.h>
  18. #include "psmouse.h"
  19. #include "trackpoint.h"
  20. static const char * const trackpoint_variants[] = {
  21. [TP_VARIANT_IBM] = "IBM",
  22. [TP_VARIANT_ALPS] = "ALPS",
  23. [TP_VARIANT_ELAN] = "Elan",
  24. [TP_VARIANT_NXP] = "NXP",
  25. };
  26. /*
  27. * Power-on Reset: Resets all trackpoint parameters, including RAM values,
  28. * to defaults.
  29. * Returns zero on success, non-zero on failure.
  30. */
  31. static int trackpoint_power_on_reset(struct ps2dev *ps2dev)
  32. {
  33. u8 param[2] = { TP_POR };
  34. int err;
  35. err = ps2_command(ps2dev, param, MAKE_PS2_CMD(1, 2, TP_COMMAND));
  36. if (err)
  37. return err;
  38. /* Check for success response -- 0xAA00 */
  39. if (param[0] != 0xAA || param[1] != 0x00)
  40. return -ENODEV;
  41. return 0;
  42. }
  43. /*
  44. * Device IO: read, write and toggle bit
  45. */
  46. static int trackpoint_read(struct ps2dev *ps2dev, u8 loc, u8 *results)
  47. {
  48. results[0] = loc;
  49. return ps2_command(ps2dev, results, MAKE_PS2_CMD(1, 1, TP_COMMAND));
  50. }
  51. static int trackpoint_write(struct ps2dev *ps2dev, u8 loc, u8 val)
  52. {
  53. u8 param[3] = { TP_WRITE_MEM, loc, val };
  54. return ps2_command(ps2dev, param, MAKE_PS2_CMD(3, 0, TP_COMMAND));
  55. }
  56. static int trackpoint_toggle_bit(struct ps2dev *ps2dev, u8 loc, u8 mask)
  57. {
  58. u8 param[3] = { TP_TOGGLE, loc, mask };
  59. /* Bad things will happen if the loc param isn't in this range */
  60. if (loc < 0x20 || loc >= 0x2F)
  61. return -EINVAL;
  62. return ps2_command(ps2dev, param, MAKE_PS2_CMD(3, 0, TP_COMMAND));
  63. }
  64. static int trackpoint_update_bit(struct ps2dev *ps2dev,
  65. u8 loc, u8 mask, u8 value)
  66. {
  67. int retval;
  68. u8 data;
  69. retval = trackpoint_read(ps2dev, loc, &data);
  70. if (retval)
  71. return retval;
  72. if (((data & mask) == mask) != !!value)
  73. retval = trackpoint_toggle_bit(ps2dev, loc, mask);
  74. return retval;
  75. }
  76. /*
  77. * Trackpoint-specific attributes
  78. */
  79. struct trackpoint_attr_data {
  80. size_t field_offset;
  81. u8 command;
  82. u8 mask;
  83. bool inverted;
  84. u8 power_on_default;
  85. };
  86. static ssize_t trackpoint_show_int_attr(struct psmouse *psmouse,
  87. void *data, char *buf)
  88. {
  89. struct trackpoint_data *tp = psmouse->private;
  90. struct trackpoint_attr_data *attr = data;
  91. u8 value = *(u8 *)((void *)tp + attr->field_offset);
  92. if (attr->inverted)
  93. value = !value;
  94. return sprintf(buf, "%u\n", value);
  95. }
  96. static ssize_t trackpoint_set_int_attr(struct psmouse *psmouse, void *data,
  97. const char *buf, size_t count)
  98. {
  99. struct trackpoint_data *tp = psmouse->private;
  100. struct trackpoint_attr_data *attr = data;
  101. u8 *field = (void *)tp + attr->field_offset;
  102. u8 value;
  103. int err;
  104. err = kstrtou8(buf, 10, &value);
  105. if (err)
  106. return err;
  107. *field = value;
  108. err = trackpoint_write(&psmouse->ps2dev, attr->command, value);
  109. return err ?: count;
  110. }
  111. #define TRACKPOINT_INT_ATTR(_name, _command, _default) \
  112. static struct trackpoint_attr_data trackpoint_attr_##_name = { \
  113. .field_offset = offsetof(struct trackpoint_data, _name), \
  114. .command = _command, \
  115. .power_on_default = _default, \
  116. }; \
  117. PSMOUSE_DEFINE_ATTR(_name, S_IWUSR | S_IRUGO, \
  118. &trackpoint_attr_##_name, \
  119. trackpoint_show_int_attr, trackpoint_set_int_attr)
  120. static ssize_t trackpoint_set_bit_attr(struct psmouse *psmouse, void *data,
  121. const char *buf, size_t count)
  122. {
  123. struct trackpoint_data *tp = psmouse->private;
  124. struct trackpoint_attr_data *attr = data;
  125. bool *field = (void *)tp + attr->field_offset;
  126. bool value;
  127. int err;
  128. err = kstrtobool(buf, &value);
  129. if (err)
  130. return err;
  131. if (attr->inverted)
  132. value = !value;
  133. if (*field != value) {
  134. *field = value;
  135. err = trackpoint_toggle_bit(&psmouse->ps2dev,
  136. attr->command, attr->mask);
  137. }
  138. return err ?: count;
  139. }
  140. #define TRACKPOINT_BIT_ATTR(_name, _command, _mask, _inv, _default) \
  141. static struct trackpoint_attr_data trackpoint_attr_##_name = { \
  142. .field_offset = offsetof(struct trackpoint_data, \
  143. _name), \
  144. .command = _command, \
  145. .mask = _mask, \
  146. .inverted = _inv, \
  147. .power_on_default = _default, \
  148. }; \
  149. PSMOUSE_DEFINE_ATTR(_name, S_IWUSR | S_IRUGO, \
  150. &trackpoint_attr_##_name, \
  151. trackpoint_show_int_attr, trackpoint_set_bit_attr)
  152. TRACKPOINT_INT_ATTR(sensitivity, TP_SENS, TP_DEF_SENS);
  153. TRACKPOINT_INT_ATTR(speed, TP_SPEED, TP_DEF_SPEED);
  154. TRACKPOINT_INT_ATTR(inertia, TP_INERTIA, TP_DEF_INERTIA);
  155. TRACKPOINT_INT_ATTR(reach, TP_REACH, TP_DEF_REACH);
  156. TRACKPOINT_INT_ATTR(draghys, TP_DRAGHYS, TP_DEF_DRAGHYS);
  157. TRACKPOINT_INT_ATTR(mindrag, TP_MINDRAG, TP_DEF_MINDRAG);
  158. TRACKPOINT_INT_ATTR(thresh, TP_THRESH, TP_DEF_THRESH);
  159. TRACKPOINT_INT_ATTR(upthresh, TP_UP_THRESH, TP_DEF_UP_THRESH);
  160. TRACKPOINT_INT_ATTR(ztime, TP_Z_TIME, TP_DEF_Z_TIME);
  161. TRACKPOINT_INT_ATTR(jenks, TP_JENKS_CURV, TP_DEF_JENKS_CURV);
  162. TRACKPOINT_INT_ATTR(drift_time, TP_DRIFT_TIME, TP_DEF_DRIFT_TIME);
  163. TRACKPOINT_BIT_ATTR(press_to_select, TP_TOGGLE_PTSON, TP_MASK_PTSON, false,
  164. TP_DEF_PTSON);
  165. TRACKPOINT_BIT_ATTR(skipback, TP_TOGGLE_SKIPBACK, TP_MASK_SKIPBACK, false,
  166. TP_DEF_SKIPBACK);
  167. TRACKPOINT_BIT_ATTR(ext_dev, TP_TOGGLE_EXT_DEV, TP_MASK_EXT_DEV, true,
  168. TP_DEF_EXT_DEV);
  169. static bool trackpoint_is_attr_available(struct psmouse *psmouse,
  170. struct attribute *attr)
  171. {
  172. struct trackpoint_data *tp = psmouse->private;
  173. return tp->variant_id == TP_VARIANT_IBM ||
  174. attr == &psmouse_attr_sensitivity.dattr.attr ||
  175. attr == &psmouse_attr_press_to_select.dattr.attr;
  176. }
  177. static umode_t trackpoint_is_attr_visible(struct kobject *kobj,
  178. struct attribute *attr, int n)
  179. {
  180. struct device *dev = container_of(kobj, struct device, kobj);
  181. struct serio *serio = to_serio_port(dev);
  182. struct psmouse *psmouse = serio_get_drvdata(serio);
  183. return trackpoint_is_attr_available(psmouse, attr) ? attr->mode : 0;
  184. }
  185. static struct attribute *trackpoint_attrs[] = {
  186. &psmouse_attr_sensitivity.dattr.attr,
  187. &psmouse_attr_speed.dattr.attr,
  188. &psmouse_attr_inertia.dattr.attr,
  189. &psmouse_attr_reach.dattr.attr,
  190. &psmouse_attr_draghys.dattr.attr,
  191. &psmouse_attr_mindrag.dattr.attr,
  192. &psmouse_attr_thresh.dattr.attr,
  193. &psmouse_attr_upthresh.dattr.attr,
  194. &psmouse_attr_ztime.dattr.attr,
  195. &psmouse_attr_jenks.dattr.attr,
  196. &psmouse_attr_drift_time.dattr.attr,
  197. &psmouse_attr_press_to_select.dattr.attr,
  198. &psmouse_attr_skipback.dattr.attr,
  199. &psmouse_attr_ext_dev.dattr.attr,
  200. NULL
  201. };
  202. static struct attribute_group trackpoint_attr_group = {
  203. .is_visible = trackpoint_is_attr_visible,
  204. .attrs = trackpoint_attrs,
  205. };
  206. #define TRACKPOINT_UPDATE(_power_on, _psmouse, _tp, _name) \
  207. do { \
  208. struct trackpoint_attr_data *_attr = &trackpoint_attr_##_name; \
  209. \
  210. if ((!_power_on || _tp->_name != _attr->power_on_default) && \
  211. trackpoint_is_attr_available(_psmouse, \
  212. &psmouse_attr_##_name.dattr.attr)) { \
  213. if (!_attr->mask) \
  214. trackpoint_write(&_psmouse->ps2dev, \
  215. _attr->command, _tp->_name); \
  216. else \
  217. trackpoint_update_bit(&_psmouse->ps2dev, \
  218. _attr->command, _attr->mask, \
  219. _tp->_name); \
  220. } \
  221. } while (0)
  222. #define TRACKPOINT_SET_POWER_ON_DEFAULT(_tp, _name) \
  223. do { \
  224. _tp->_name = trackpoint_attr_##_name.power_on_default; \
  225. } while (0)
  226. static int trackpoint_start_protocol(struct psmouse *psmouse,
  227. u8 *variant_id, u8 *firmware_id)
  228. {
  229. u8 param[2] = { 0 };
  230. int error;
  231. error = ps2_command(&psmouse->ps2dev,
  232. param, MAKE_PS2_CMD(0, 2, TP_READ_ID));
  233. if (error)
  234. return error;
  235. switch (param[0]) {
  236. case TP_VARIANT_IBM:
  237. case TP_VARIANT_ALPS:
  238. case TP_VARIANT_ELAN:
  239. case TP_VARIANT_NXP:
  240. if (variant_id)
  241. *variant_id = param[0];
  242. if (firmware_id)
  243. *firmware_id = param[1];
  244. return 0;
  245. }
  246. return -ENODEV;
  247. }
  248. /*
  249. * Write parameters to trackpad.
  250. * in_power_on_state: Set to true if TP is in default / power-on state (ex. if
  251. * power-on reset was run). If so, values will only be
  252. * written to TP if they differ from power-on default.
  253. */
  254. static int trackpoint_sync(struct psmouse *psmouse, bool in_power_on_state)
  255. {
  256. struct trackpoint_data *tp = psmouse->private;
  257. if (!in_power_on_state && tp->variant_id == TP_VARIANT_IBM) {
  258. /*
  259. * Disable features that may make device unusable
  260. * with this driver.
  261. */
  262. trackpoint_update_bit(&psmouse->ps2dev, TP_TOGGLE_TWOHAND,
  263. TP_MASK_TWOHAND, TP_DEF_TWOHAND);
  264. trackpoint_update_bit(&psmouse->ps2dev, TP_TOGGLE_SOURCE_TAG,
  265. TP_MASK_SOURCE_TAG, TP_DEF_SOURCE_TAG);
  266. trackpoint_update_bit(&psmouse->ps2dev, TP_TOGGLE_MB,
  267. TP_MASK_MB, TP_DEF_MB);
  268. }
  269. /*
  270. * These properties can be changed in this driver. Only
  271. * configure them if the values are non-default or if the TP is in
  272. * an unknown state.
  273. */
  274. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, sensitivity);
  275. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, inertia);
  276. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, speed);
  277. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, reach);
  278. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, draghys);
  279. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, mindrag);
  280. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, thresh);
  281. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, upthresh);
  282. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, ztime);
  283. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, jenks);
  284. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, drift_time);
  285. /* toggles */
  286. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, press_to_select);
  287. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, skipback);
  288. TRACKPOINT_UPDATE(in_power_on_state, psmouse, tp, ext_dev);
  289. return 0;
  290. }
  291. static void trackpoint_defaults(struct trackpoint_data *tp)
  292. {
  293. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, sensitivity);
  294. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, speed);
  295. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, reach);
  296. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, draghys);
  297. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, mindrag);
  298. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, thresh);
  299. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, upthresh);
  300. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, ztime);
  301. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, jenks);
  302. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, drift_time);
  303. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, inertia);
  304. /* toggles */
  305. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, press_to_select);
  306. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, skipback);
  307. TRACKPOINT_SET_POWER_ON_DEFAULT(tp, ext_dev);
  308. }
  309. static void trackpoint_disconnect(struct psmouse *psmouse)
  310. {
  311. device_remove_group(&psmouse->ps2dev.serio->dev,
  312. &trackpoint_attr_group);
  313. kfree(psmouse->private);
  314. psmouse->private = NULL;
  315. }
  316. static int trackpoint_reconnect(struct psmouse *psmouse)
  317. {
  318. struct trackpoint_data *tp = psmouse->private;
  319. int error;
  320. bool was_reset;
  321. error = trackpoint_start_protocol(psmouse, NULL, NULL);
  322. if (error)
  323. return error;
  324. was_reset = tp->variant_id == TP_VARIANT_IBM &&
  325. trackpoint_power_on_reset(&psmouse->ps2dev) == 0;
  326. error = trackpoint_sync(psmouse, was_reset);
  327. if (error)
  328. return error;
  329. return 0;
  330. }
  331. int trackpoint_detect(struct psmouse *psmouse, bool set_properties)
  332. {
  333. struct ps2dev *ps2dev = &psmouse->ps2dev;
  334. struct trackpoint_data *tp;
  335. u8 variant_id;
  336. u8 firmware_id;
  337. u8 button_info;
  338. int error;
  339. error = trackpoint_start_protocol(psmouse, &variant_id, &firmware_id);
  340. if (error)
  341. return error;
  342. if (!set_properties)
  343. return 0;
  344. tp = kzalloc(sizeof(*tp), GFP_KERNEL);
  345. if (!tp)
  346. return -ENOMEM;
  347. trackpoint_defaults(tp);
  348. tp->variant_id = variant_id;
  349. tp->firmware_id = firmware_id;
  350. psmouse->private = tp;
  351. psmouse->vendor = trackpoint_variants[variant_id];
  352. psmouse->name = "TrackPoint";
  353. psmouse->reconnect = trackpoint_reconnect;
  354. psmouse->disconnect = trackpoint_disconnect;
  355. if (variant_id != TP_VARIANT_IBM) {
  356. /* Newer variants do not support extended button query. */
  357. button_info = 0x33;
  358. } else {
  359. error = trackpoint_read(ps2dev, TP_EXT_BTN, &button_info);
  360. if (error) {
  361. psmouse_warn(psmouse,
  362. "failed to get extended button data, assuming 3 buttons\n");
  363. button_info = 0x33;
  364. } else if (!button_info) {
  365. psmouse_warn(psmouse,
  366. "got 0 in extended button data, assuming 3 buttons\n");
  367. button_info = 0x33;
  368. }
  369. }
  370. if ((button_info & 0x0f) >= 3)
  371. input_set_capability(psmouse->dev, EV_KEY, BTN_MIDDLE);
  372. __set_bit(INPUT_PROP_POINTER, psmouse->dev->propbit);
  373. __set_bit(INPUT_PROP_POINTING_STICK, psmouse->dev->propbit);
  374. if (variant_id != TP_VARIANT_IBM ||
  375. trackpoint_power_on_reset(ps2dev) != 0) {
  376. /*
  377. * Write defaults to TP if we did not reset the trackpoint.
  378. */
  379. trackpoint_sync(psmouse, false);
  380. }
  381. error = device_add_group(&ps2dev->serio->dev, &trackpoint_attr_group);
  382. if (error) {
  383. psmouse_err(psmouse,
  384. "failed to create sysfs attributes, error: %d\n",
  385. error);
  386. kfree(psmouse->private);
  387. psmouse->private = NULL;
  388. return -1;
  389. }
  390. psmouse_info(psmouse,
  391. "%s TrackPoint firmware: 0x%02x, buttons: %d/%d\n",
  392. psmouse->vendor, firmware_id,
  393. (button_info & 0xf0) >> 4, button_info & 0x0f);
  394. return 0;
  395. }