diff options
-rw-r--r-- | drivers/platform/x86/sony-laptop.c | 71 |
1 files changed, 39 insertions, 32 deletions
diff --git a/drivers/platform/x86/sony-laptop.c b/drivers/platform/x86/sony-laptop.c index 5af53340da6f..c42d35ba73d6 100644 --- a/drivers/platform/x86/sony-laptop.c +++ b/drivers/platform/x86/sony-laptop.c | |||
@@ -145,7 +145,6 @@ struct sony_laptop_input_s { | |||
145 | struct input_dev *key_dev; | 145 | struct input_dev *key_dev; |
146 | struct kfifo fifo; | 146 | struct kfifo fifo; |
147 | spinlock_t fifo_lock; | 147 | spinlock_t fifo_lock; |
148 | struct workqueue_struct *wq; | ||
149 | }; | 148 | }; |
150 | 149 | ||
151 | static struct sony_laptop_input_s sony_laptop_input = { | 150 | static struct sony_laptop_input_s sony_laptop_input = { |
@@ -301,18 +300,28 @@ static int sony_laptop_input_keycode_map[] = { | |||
301 | /* release buttons after a short delay if pressed */ | 300 | /* release buttons after a short delay if pressed */ |
302 | static void do_sony_laptop_release_key(struct work_struct *work) | 301 | static void do_sony_laptop_release_key(struct work_struct *work) |
303 | { | 302 | { |
303 | struct delayed_work *dwork = | ||
304 | container_of(work, struct delayed_work, work); | ||
304 | struct sony_laptop_keypress kp; | 305 | struct sony_laptop_keypress kp; |
306 | unsigned long flags; | ||
307 | |||
308 | spin_lock_irqsave(&sony_laptop_input.fifo_lock, flags); | ||
305 | 309 | ||
306 | while (kfifo_out_locked(&sony_laptop_input.fifo, (unsigned char *)&kp, | 310 | if (kfifo_out(&sony_laptop_input.fifo, |
307 | sizeof(kp), &sony_laptop_input.fifo_lock) | 311 | (unsigned char *)&kp, sizeof(kp)) == sizeof(kp)) { |
308 | == sizeof(kp)) { | ||
309 | msleep(10); | ||
310 | input_report_key(kp.dev, kp.key, 0); | 312 | input_report_key(kp.dev, kp.key, 0); |
311 | input_sync(kp.dev); | 313 | input_sync(kp.dev); |
312 | } | 314 | } |
315 | |||
316 | /* If there is something in the fifo schedule next release. */ | ||
317 | if (kfifo_len(&sony_laptop_input.fifo) != 0) | ||
318 | schedule_delayed_work(dwork, msecs_to_jiffies(10)); | ||
319 | |||
320 | spin_unlock_irqrestore(&sony_laptop_input.fifo_lock, flags); | ||
313 | } | 321 | } |
314 | static DECLARE_WORK(sony_laptop_release_key_work, | 322 | |
315 | do_sony_laptop_release_key); | 323 | static DECLARE_DELAYED_WORK(sony_laptop_release_key_work, |
324 | do_sony_laptop_release_key); | ||
316 | 325 | ||
317 | /* forward event to the input subsystem */ | 326 | /* forward event to the input subsystem */ |
318 | static void sony_laptop_report_input_event(u8 event) | 327 | static void sony_laptop_report_input_event(u8 event) |
@@ -366,13 +375,13 @@ static void sony_laptop_report_input_event(u8 event) | |||
366 | /* we emit the scancode so we can always remap the key */ | 375 | /* we emit the scancode so we can always remap the key */ |
367 | input_event(kp.dev, EV_MSC, MSC_SCAN, event); | 376 | input_event(kp.dev, EV_MSC, MSC_SCAN, event); |
368 | input_sync(kp.dev); | 377 | input_sync(kp.dev); |
369 | kfifo_in_locked(&sony_laptop_input.fifo, | ||
370 | (unsigned char *)&kp, sizeof(kp), | ||
371 | &sony_laptop_input.fifo_lock); | ||
372 | 378 | ||
373 | if (!work_pending(&sony_laptop_release_key_work)) | 379 | /* schedule key release */ |
374 | queue_work(sony_laptop_input.wq, | 380 | kfifo_in_locked(&sony_laptop_input.fifo, |
375 | &sony_laptop_release_key_work); | 381 | (unsigned char *)&kp, sizeof(kp), |
382 | &sony_laptop_input.fifo_lock); | ||
383 | schedule_delayed_work(&sony_laptop_release_key_work, | ||
384 | msecs_to_jiffies(10)); | ||
376 | } else | 385 | } else |
377 | dprintk("unknown input event %.2x\n", event); | 386 | dprintk("unknown input event %.2x\n", event); |
378 | } | 387 | } |
@@ -390,27 +399,18 @@ static int sony_laptop_setup_input(struct acpi_device *acpi_device) | |||
390 | 399 | ||
391 | /* kfifo */ | 400 | /* kfifo */ |
392 | spin_lock_init(&sony_laptop_input.fifo_lock); | 401 | spin_lock_init(&sony_laptop_input.fifo_lock); |
393 | error = | 402 | error = kfifo_alloc(&sony_laptop_input.fifo, |
394 | kfifo_alloc(&sony_laptop_input.fifo, SONY_LAPTOP_BUF_SIZE, GFP_KERNEL); | 403 | SONY_LAPTOP_BUF_SIZE, GFP_KERNEL); |
395 | if (error) { | 404 | if (error) { |
396 | printk(KERN_ERR DRV_PFX "kfifo_alloc failed\n"); | 405 | printk(KERN_ERR DRV_PFX "kfifo_alloc failed\n"); |
397 | goto err_dec_users; | 406 | goto err_dec_users; |
398 | } | 407 | } |
399 | 408 | ||
400 | /* init workqueue */ | ||
401 | sony_laptop_input.wq = create_singlethread_workqueue("sony-laptop"); | ||
402 | if (!sony_laptop_input.wq) { | ||
403 | printk(KERN_ERR DRV_PFX | ||
404 | "Unable to create workqueue.\n"); | ||
405 | error = -ENXIO; | ||
406 | goto err_free_kfifo; | ||
407 | } | ||
408 | |||
409 | /* input keys */ | 409 | /* input keys */ |
410 | key_dev = input_allocate_device(); | 410 | key_dev = input_allocate_device(); |
411 | if (!key_dev) { | 411 | if (!key_dev) { |
412 | error = -ENOMEM; | 412 | error = -ENOMEM; |
413 | goto err_destroy_wq; | 413 | goto err_free_kfifo; |
414 | } | 414 | } |
415 | 415 | ||
416 | key_dev->name = "Sony Vaio Keys"; | 416 | key_dev->name = "Sony Vaio Keys"; |
@@ -473,9 +473,6 @@ err_unregister_keydev: | |||
473 | err_free_keydev: | 473 | err_free_keydev: |
474 | input_free_device(key_dev); | 474 | input_free_device(key_dev); |
475 | 475 | ||
476 | err_destroy_wq: | ||
477 | destroy_workqueue(sony_laptop_input.wq); | ||
478 | |||
479 | err_free_kfifo: | 476 | err_free_kfifo: |
480 | kfifo_free(&sony_laptop_input.fifo); | 477 | kfifo_free(&sony_laptop_input.fifo); |
481 | 478 | ||
@@ -486,12 +483,23 @@ err_dec_users: | |||
486 | 483 | ||
487 | static void sony_laptop_remove_input(void) | 484 | static void sony_laptop_remove_input(void) |
488 | { | 485 | { |
489 | /* cleanup only after the last user has gone */ | 486 | struct sony_laptop_keypress kp = { NULL }; |
487 | |||
488 | /* Cleanup only after the last user has gone */ | ||
490 | if (!atomic_dec_and_test(&sony_laptop_input.users)) | 489 | if (!atomic_dec_and_test(&sony_laptop_input.users)) |
491 | return; | 490 | return; |
492 | 491 | ||
493 | /* flush workqueue first */ | 492 | cancel_delayed_work_sync(&sony_laptop_release_key_work); |
494 | flush_workqueue(sony_laptop_input.wq); | 493 | |
494 | /* | ||
495 | * Generate key-up events for remaining keys. Note that we don't | ||
496 | * need locking since nobody is adding new events to the kfifo. | ||
497 | */ | ||
498 | while (kfifo_out(&sony_laptop_input.fifo, | ||
499 | (unsigned char *)&kp, sizeof(kp)) == sizeof(kp)) { | ||
500 | input_report_key(kp.dev, kp.key, 0); | ||
501 | input_sync(kp.dev); | ||
502 | } | ||
495 | 503 | ||
496 | /* destroy input devs */ | 504 | /* destroy input devs */ |
497 | input_unregister_device(sony_laptop_input.key_dev); | 505 | input_unregister_device(sony_laptop_input.key_dev); |
@@ -502,7 +510,6 @@ static void sony_laptop_remove_input(void) | |||
502 | sony_laptop_input.jog_dev = NULL; | 510 | sony_laptop_input.jog_dev = NULL; |
503 | } | 511 | } |
504 | 512 | ||
505 | destroy_workqueue(sony_laptop_input.wq); | ||
506 | kfifo_free(&sony_laptop_input.fifo); | 513 | kfifo_free(&sony_laptop_input.fifo); |
507 | } | 514 | } |
508 | 515 | ||