浏览代码

Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/dtor/input

* 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/dtor/input:
  Input: fix locking in force-feedback core
  Input: add KEY_MEDIA_REPEAT definition
Linus Torvalds 17 年之前
父节点
当前提交
e1441b9a41
共有 2 个文件被更改,包括 12 次插入7 次删除
  1. 11 6
      drivers/input/ff-core.c
  2. 1 1
      include/linux/input.h

+ 11 - 6
drivers/input/ff-core.c

@@ -166,8 +166,10 @@ int input_ff_upload(struct input_dev *dev, struct ff_effect *effect,
 	if (ret)
 	if (ret)
 		goto out;
 		goto out;
 
 
+	spin_lock_irq(&dev->event_lock);
 	ff->effects[id] = *effect;
 	ff->effects[id] = *effect;
 	ff->effect_owners[id] = file;
 	ff->effect_owners[id] = file;
+	spin_unlock_irq(&dev->event_lock);
 
 
  out:
  out:
 	mutex_unlock(&ff->mutex);
 	mutex_unlock(&ff->mutex);
@@ -189,16 +191,22 @@ static int erase_effect(struct input_dev *dev, int effect_id,
 	if (error)
 	if (error)
 		return error;
 		return error;
 
 
+	spin_lock_irq(&dev->event_lock);
 	ff->playback(dev, effect_id, 0);
 	ff->playback(dev, effect_id, 0);
+	ff->effect_owners[effect_id] = NULL;
+	spin_unlock_irq(&dev->event_lock);
 
 
 	if (ff->erase) {
 	if (ff->erase) {
 		error = ff->erase(dev, effect_id);
 		error = ff->erase(dev, effect_id);
-		if (error)
+		if (error) {
+			spin_lock_irq(&dev->event_lock);
+			ff->effect_owners[effect_id] = file;
+			spin_unlock_irq(&dev->event_lock);
+
 			return error;
 			return error;
+		}
 	}
 	}
 
 
-	ff->effect_owners[effect_id] = NULL;
-
 	return 0;
 	return 0;
 }
 }
 
 
@@ -263,8 +271,6 @@ int input_ff_event(struct input_dev *dev, unsigned int type,
 	if (type != EV_FF)
 	if (type != EV_FF)
 		return 0;
 		return 0;
 
 
-	mutex_lock(&ff->mutex);
-
 	switch (code) {
 	switch (code) {
 	case FF_GAIN:
 	case FF_GAIN:
 		if (!test_bit(FF_GAIN, dev->ffbit) || value > 0xffff)
 		if (!test_bit(FF_GAIN, dev->ffbit) || value > 0xffff)
@@ -286,7 +292,6 @@ int input_ff_event(struct input_dev *dev, unsigned int type,
 		break;
 		break;
 	}
 	}
 
 
-	mutex_unlock(&ff->mutex);
 	return 0;
 	return 0;
 }
 }
 EXPORT_SYMBOL_GPL(input_ff_event);
 EXPORT_SYMBOL_GPL(input_ff_event);

+ 1 - 1
include/linux/input.h

@@ -534,8 +534,8 @@ struct input_absinfo {
 
 
 #define KEY_FRAMEBACK		0x1b4	/* Consumer - transport controls */
 #define KEY_FRAMEBACK		0x1b4	/* Consumer - transport controls */
 #define KEY_FRAMEFORWARD	0x1b5
 #define KEY_FRAMEFORWARD	0x1b5
-
 #define KEY_CONTEXT_MENU	0x1b6	/* GenDesc - system context menu */
 #define KEY_CONTEXT_MENU	0x1b6	/* GenDesc - system context menu */
+#define KEY_MEDIA_REPEAT	0x1b7	/* Consumer - transport control */
 
 
 #define KEY_DEL_EOL		0x1c0
 #define KEY_DEL_EOL		0x1c0
 #define KEY_DEL_EOS		0x1c1
 #define KEY_DEL_EOS		0x1c1