diff options
Diffstat (limited to 'drivers/input/input.c')
| -rw-r--r-- | drivers/input/input.c | 94 |
1 files changed, 66 insertions, 28 deletions
diff --git a/drivers/input/input.c b/drivers/input/input.c index c6f88ebb40c7..2266ecbfbc01 100644 --- a/drivers/input/input.c +++ b/drivers/input/input.c | |||
| @@ -782,10 +782,29 @@ static unsigned int input_proc_devices_poll(struct file *file, poll_table *wait) | |||
| 782 | return 0; | 782 | return 0; |
| 783 | } | 783 | } |
| 784 | 784 | ||
| 785 | union input_seq_state { | ||
| 786 | struct { | ||
| 787 | unsigned short pos; | ||
| 788 | bool mutex_acquired; | ||
| 789 | }; | ||
| 790 | void *p; | ||
| 791 | }; | ||
| 792 | |||
| 785 | static void *input_devices_seq_start(struct seq_file *seq, loff_t *pos) | 793 | static void *input_devices_seq_start(struct seq_file *seq, loff_t *pos) |
| 786 | { | 794 | { |
| 787 | if (mutex_lock_interruptible(&input_mutex)) | 795 | union input_seq_state *state = (union input_seq_state *)&seq->private; |
| 788 | return NULL; | 796 | int error; |
| 797 | |||
| 798 | /* We need to fit into seq->private pointer */ | ||
| 799 | BUILD_BUG_ON(sizeof(union input_seq_state) != sizeof(seq->private)); | ||
| 800 | |||
| 801 | error = mutex_lock_interruptible(&input_mutex); | ||
| 802 | if (error) { | ||
| 803 | state->mutex_acquired = false; | ||
| 804 | return ERR_PTR(error); | ||
| 805 | } | ||
| 806 | |||
| 807 | state->mutex_acquired = true; | ||
| 789 | 808 | ||
| 790 | return seq_list_start(&input_dev_list, *pos); | 809 | return seq_list_start(&input_dev_list, *pos); |
| 791 | } | 810 | } |
| @@ -795,9 +814,12 @@ static void *input_devices_seq_next(struct seq_file *seq, void *v, loff_t *pos) | |||
| 795 | return seq_list_next(v, &input_dev_list, pos); | 814 | return seq_list_next(v, &input_dev_list, pos); |
| 796 | } | 815 | } |
| 797 | 816 | ||
| 798 | static void input_devices_seq_stop(struct seq_file *seq, void *v) | 817 | static void input_seq_stop(struct seq_file *seq, void *v) |
| 799 | { | 818 | { |
| 800 | mutex_unlock(&input_mutex); | 819 | union input_seq_state *state = (union input_seq_state *)&seq->private; |
| 820 | |||
| 821 | if (state->mutex_acquired) | ||
| 822 | mutex_unlock(&input_mutex); | ||
| 801 | } | 823 | } |
| 802 | 824 | ||
| 803 | static void input_seq_print_bitmap(struct seq_file *seq, const char *name, | 825 | static void input_seq_print_bitmap(struct seq_file *seq, const char *name, |
| @@ -861,7 +883,7 @@ static int input_devices_seq_show(struct seq_file *seq, void *v) | |||
| 861 | static const struct seq_operations input_devices_seq_ops = { | 883 | static const struct seq_operations input_devices_seq_ops = { |
| 862 | .start = input_devices_seq_start, | 884 | .start = input_devices_seq_start, |
| 863 | .next = input_devices_seq_next, | 885 | .next = input_devices_seq_next, |
| 864 | .stop = input_devices_seq_stop, | 886 | .stop = input_seq_stop, |
| 865 | .show = input_devices_seq_show, | 887 | .show = input_devices_seq_show, |
| 866 | }; | 888 | }; |
| 867 | 889 | ||
| @@ -881,40 +903,49 @@ static const struct file_operations input_devices_fileops = { | |||
| 881 | 903 | ||
| 882 | static void *input_handlers_seq_start(struct seq_file *seq, loff_t *pos) | 904 | static void *input_handlers_seq_start(struct seq_file *seq, loff_t *pos) |
| 883 | { | 905 | { |
| 884 | if (mutex_lock_interruptible(&input_mutex)) | 906 | union input_seq_state *state = (union input_seq_state *)&seq->private; |
| 885 | return NULL; | 907 | int error; |
| 908 | |||
| 909 | /* We need to fit into seq->private pointer */ | ||
| 910 | BUILD_BUG_ON(sizeof(union input_seq_state) != sizeof(seq->private)); | ||
| 911 | |||
| 912 | error = mutex_lock_interruptible(&input_mutex); | ||
| 913 | if (error) { | ||
| 914 | state->mutex_acquired = false; | ||
| 915 | return ERR_PTR(error); | ||
| 916 | } | ||
| 917 | |||
| 918 | state->mutex_acquired = true; | ||
| 919 | state->pos = *pos; | ||
| 886 | 920 | ||
| 887 | seq->private = (void *)(unsigned long)*pos; | ||
| 888 | return seq_list_start(&input_handler_list, *pos); | 921 | return seq_list_start(&input_handler_list, *pos); |
| 889 | } | 922 | } |
| 890 | 923 | ||
| 891 | static void *input_handlers_seq_next(struct seq_file *seq, void *v, loff_t *pos) | 924 | static void *input_handlers_seq_next(struct seq_file *seq, void *v, loff_t *pos) |
| 892 | { | 925 | { |
| 893 | seq->private = (void *)(unsigned long)(*pos + 1); | 926 | union input_seq_state *state = (union input_seq_state *)&seq->private; |
| 894 | return seq_list_next(v, &input_handler_list, pos); | ||
| 895 | } | ||
| 896 | 927 | ||
| 897 | static void input_handlers_seq_stop(struct seq_file *seq, void *v) | 928 | state->pos = *pos + 1; |
| 898 | { | 929 | return seq_list_next(v, &input_handler_list, pos); |
| 899 | mutex_unlock(&input_mutex); | ||
| 900 | } | 930 | } |
| 901 | 931 | ||
| 902 | static int input_handlers_seq_show(struct seq_file *seq, void *v) | 932 | static int input_handlers_seq_show(struct seq_file *seq, void *v) |
| 903 | { | 933 | { |
| 904 | struct input_handler *handler = container_of(v, struct input_handler, node); | 934 | struct input_handler *handler = container_of(v, struct input_handler, node); |
| 935 | union input_seq_state *state = (union input_seq_state *)&seq->private; | ||
| 905 | 936 | ||
| 906 | seq_printf(seq, "N: Number=%ld Name=%s", | 937 | seq_printf(seq, "N: Number=%u Name=%s", state->pos, handler->name); |
| 907 | (unsigned long)seq->private, handler->name); | ||
| 908 | if (handler->fops) | 938 | if (handler->fops) |
| 909 | seq_printf(seq, " Minor=%d", handler->minor); | 939 | seq_printf(seq, " Minor=%d", handler->minor); |
| 910 | seq_putc(seq, '\n'); | 940 | seq_putc(seq, '\n'); |
| 911 | 941 | ||
| 912 | return 0; | 942 | return 0; |
| 913 | } | 943 | } |
| 944 | |||
| 914 | static const struct seq_operations input_handlers_seq_ops = { | 945 | static const struct seq_operations input_handlers_seq_ops = { |
| 915 | .start = input_handlers_seq_start, | 946 | .start = input_handlers_seq_start, |
| 916 | .next = input_handlers_seq_next, | 947 | .next = input_handlers_seq_next, |
| 917 | .stop = input_handlers_seq_stop, | 948 | .stop = input_seq_stop, |
| 918 | .show = input_handlers_seq_show, | 949 | .show = input_handlers_seq_show, |
| 919 | }; | 950 | }; |
| 920 | 951 | ||
| @@ -1261,17 +1292,24 @@ static int input_dev_uevent(struct device *device, struct kobj_uevent_env *env) | |||
| 1261 | return 0; | 1292 | return 0; |
| 1262 | } | 1293 | } |
| 1263 | 1294 | ||
| 1264 | #define INPUT_DO_TOGGLE(dev, type, bits, on) \ | 1295 | #define INPUT_DO_TOGGLE(dev, type, bits, on) \ |
| 1265 | do { \ | 1296 | do { \ |
| 1266 | int i; \ | 1297 | int i; \ |
| 1267 | if (!test_bit(EV_##type, dev->evbit)) \ | 1298 | bool active; \ |
| 1268 | break; \ | 1299 | \ |
| 1269 | for (i = 0; i < type##_MAX; i++) { \ | 1300 | if (!test_bit(EV_##type, dev->evbit)) \ |
| 1270 | if (!test_bit(i, dev->bits##bit) || \ | 1301 | break; \ |
| 1271 | !test_bit(i, dev->bits)) \ | 1302 | \ |
| 1272 | continue; \ | 1303 | for (i = 0; i < type##_MAX; i++) { \ |
| 1273 | dev->event(dev, EV_##type, i, on); \ | 1304 | if (!test_bit(i, dev->bits##bit)) \ |
| 1274 | } \ | 1305 | continue; \ |
| 1306 | \ | ||
| 1307 | active = test_bit(i, dev->bits); \ | ||
| 1308 | if (!active && !on) \ | ||
| 1309 | continue; \ | ||
| 1310 | \ | ||
| 1311 | dev->event(dev, EV_##type, i, on ? active : 0); \ | ||
| 1312 | } \ | ||
| 1275 | } while (0) | 1313 | } while (0) |
| 1276 | 1314 | ||
| 1277 | #ifdef CONFIG_PM | 1315 | #ifdef CONFIG_PM |
