diff options
Diffstat (limited to 'drivers/input/input.c')
| -rw-r--r-- | drivers/input/input.c | 65 |
1 files changed, 48 insertions, 17 deletions
diff --git a/drivers/input/input.c b/drivers/input/input.c index c6f88ebb40c7..cc763c96fada 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 | ||
