diff options
| -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 b8ed4294fccd..60a4eaabb7d7 100644 --- a/drivers/input/input.c +++ b/drivers/input/input.c | |||
| @@ -781,10 +781,29 @@ static unsigned int input_proc_devices_poll(struct file *file, poll_table *wait) | |||
| 781 | return 0; | 781 | return 0; |
| 782 | } | 782 | } |
| 783 | 783 | ||
| 784 | union input_seq_state { | ||
| 785 | struct { | ||
| 786 | unsigned short pos; | ||
| 787 | bool mutex_acquired; | ||
| 788 | }; | ||
| 789 | void *p; | ||
| 790 | }; | ||
| 791 | |||
| 784 | static void *input_devices_seq_start(struct seq_file *seq, loff_t *pos) | 792 | static void *input_devices_seq_start(struct seq_file *seq, loff_t *pos) |
| 785 | { | 793 | { |
| 786 | if (mutex_lock_interruptible(&input_mutex)) | 794 | union input_seq_state *state = (union input_seq_state *)&seq->private; |
| 787 | return NULL; | 795 | int error; |
| 796 | |||
| 797 | /* We need to fit into seq->private pointer */ | ||
| 798 | BUILD_BUG_ON(sizeof(union input_seq_state) != sizeof(seq->private)); | ||
| 799 | |||
| 800 | error = mutex_lock_interruptible(&input_mutex); | ||
| 801 | if (error) { | ||
| 802 | state->mutex_acquired = false; | ||
| 803 | return ERR_PTR(error); | ||
| 804 | } | ||
| 805 | |||
| 806 | state->mutex_acquired = true; | ||
| 788 | 807 | ||
| 789 | return seq_list_start(&input_dev_list, *pos); | 808 | return seq_list_start(&input_dev_list, *pos); |
| 790 | } | 809 | } |
| @@ -794,9 +813,12 @@ static void *input_devices_seq_next(struct seq_file *seq, void *v, loff_t *pos) | |||
| 794 | return seq_list_next(v, &input_dev_list, pos); | 813 | return seq_list_next(v, &input_dev_list, pos); |
| 795 | } | 814 | } |
| 796 | 815 | ||
| 797 | static void input_devices_seq_stop(struct seq_file *seq, void *v) | 816 | static void input_seq_stop(struct seq_file *seq, void *v) |
| 798 | { | 817 | { |
| 799 | mutex_unlock(&input_mutex); | 818 | union input_seq_state *state = (union input_seq_state *)&seq->private; |
| 819 | |||
| 820 | if (state->mutex_acquired) | ||
| 821 | mutex_unlock(&input_mutex); | ||
| 800 | } | 822 | } |
| 801 | 823 | ||
| 802 | static void input_seq_print_bitmap(struct seq_file *seq, const char *name, | 824 | static void input_seq_print_bitmap(struct seq_file *seq, const char *name, |
| @@ -860,7 +882,7 @@ static int input_devices_seq_show(struct seq_file *seq, void *v) | |||
| 860 | static const struct seq_operations input_devices_seq_ops = { | 882 | static const struct seq_operations input_devices_seq_ops = { |
| 861 | .start = input_devices_seq_start, | 883 | .start = input_devices_seq_start, |
| 862 | .next = input_devices_seq_next, | 884 | .next = input_devices_seq_next, |
| 863 | .stop = input_devices_seq_stop, | 885 | .stop = input_seq_stop, |
| 864 | .show = input_devices_seq_show, | 886 | .show = input_devices_seq_show, |
| 865 | }; | 887 | }; |
| 866 | 888 | ||
| @@ -880,40 +902,49 @@ static const struct file_operations input_devices_fileops = { | |||
| 880 | 902 | ||
| 881 | static void *input_handlers_seq_start(struct seq_file *seq, loff_t *pos) | 903 | static void *input_handlers_seq_start(struct seq_file *seq, loff_t *pos) |
| 882 | { | 904 | { |
| 883 | if (mutex_lock_interruptible(&input_mutex)) | 905 | union input_seq_state *state = (union input_seq_state *)&seq->private; |
| 884 | return NULL; | 906 | int error; |
| 907 | |||
| 908 | /* We need to fit into seq->private pointer */ | ||
| 909 | BUILD_BUG_ON(sizeof(union input_seq_state) != sizeof(seq->private)); | ||
| 910 | |||
| 911 | error = mutex_lock_interruptible(&input_mutex); | ||
| 912 | if (error) { | ||
| 913 | state->mutex_acquired = false; | ||
| 914 | return ERR_PTR(error); | ||
| 915 | } | ||
| 916 | |||
| 917 | state->mutex_acquired = true; | ||
| 918 | state->pos = *pos; | ||
| 885 | 919 | ||
| 886 | seq->private = (void *)(unsigned long)*pos; | ||
| 887 | return seq_list_start(&input_handler_list, *pos); | 920 | return seq_list_start(&input_handler_list, *pos); |
| 888 | } | 921 | } |
| 889 | 922 | ||
| 890 | static void *input_handlers_seq_next(struct seq_file *seq, void *v, loff_t *pos) | 923 | static void *input_handlers_seq_next(struct seq_file *seq, void *v, loff_t *pos) |
| 891 | { | 924 | { |
| 892 | seq->private = (void *)(unsigned long)(*pos + 1); | 925 | union input_seq_state *state = (union input_seq_state *)&seq->private; |
| 893 | return seq_list_next(v, &input_handler_list, pos); | ||
| 894 | } | ||
| 895 | 926 | ||
| 896 | static void input_handlers_seq_stop(struct seq_file *seq, void *v) | 927 | state->pos = *pos + 1; |
| 897 | { | 928 | return seq_list_next(v, &input_handler_list, pos); |
| 898 | mutex_unlock(&input_mutex); | ||
| 899 | } | 929 | } |
| 900 | 930 | ||
| 901 | static int input_handlers_seq_show(struct seq_file *seq, void *v) | 931 | static int input_handlers_seq_show(struct seq_file *seq, void *v) |
| 902 | { | 932 | { |
| 903 | struct input_handler *handler = container_of(v, struct input_handler, node); | 933 | struct input_handler *handler = container_of(v, struct input_handler, node); |
| 934 | union input_seq_state *state = (union input_seq_state *)&seq->private; | ||
| 904 | 935 | ||
| 905 | seq_printf(seq, "N: Number=%ld Name=%s", | 936 | seq_printf(seq, "N: Number=%u Name=%s", state->pos, handler->name); |
| 906 | (unsigned long)seq->private, handler->name); | ||
| 907 | if (handler->fops) | 937 | if (handler->fops) |
| 908 | seq_printf(seq, " Minor=%d", handler->minor); | 938 | seq_printf(seq, " Minor=%d", handler->minor); |
| 909 | seq_putc(seq, '\n'); | 939 | seq_putc(seq, '\n'); |
| 910 | 940 | ||
| 911 | return 0; | 941 | return 0; |
| 912 | } | 942 | } |
| 943 | |||
| 913 | static const struct seq_operations input_handlers_seq_ops = { | 944 | static const struct seq_operations input_handlers_seq_ops = { |
| 914 | .start = input_handlers_seq_start, | 945 | .start = input_handlers_seq_start, |
| 915 | .next = input_handlers_seq_next, | 946 | .next = input_handlers_seq_next, |
| 916 | .stop = input_handlers_seq_stop, | 947 | .stop = input_seq_stop, |
| 917 | .show = input_handlers_seq_show, | 948 | .show = input_handlers_seq_show, |
| 918 | }; | 949 | }; |
| 919 | 950 | ||
