| .. | .. |
|---|
| 25 | 25 | #include <linux/rfkill-bt.h> |
|---|
| 26 | 26 | #include <linux/rfkill-wlan.h> |
|---|
| 27 | 27 | #include <linux/wakelock.h> |
|---|
| 28 | +#include <linux/input.h> |
|---|
| 28 | 29 | #include <linux/interrupt.h> |
|---|
| 29 | 30 | #include <asm/irq.h> |
|---|
| 30 | 31 | #include <linux/suspend.h> |
|---|
| .. | .. |
|---|
| 69 | 70 | struct wake_lock bt_irq_wl; |
|---|
| 70 | 71 | struct delayed_work bt_sleep_delay_work; |
|---|
| 71 | 72 | int irq_req; |
|---|
| 73 | + bool enable_power_key; |
|---|
| 72 | 74 | }; |
|---|
| 73 | 75 | |
|---|
| 74 | 76 | static struct rfkill_rk_data *g_rfkill = NULL; |
|---|
| 77 | +static struct input_dev *power_key_dev; |
|---|
| 75 | 78 | |
|---|
| 76 | 79 | static const char bt_name[] = |
|---|
| 77 | 80 | #if defined(CONFIG_BCM4330) |
|---|
| .. | .. |
|---|
| 113 | 116 | #endif |
|---|
| 114 | 117 | ; |
|---|
| 115 | 118 | |
|---|
| 119 | +static int rfkill_rk_power_key_up(void) |
|---|
| 120 | +{ |
|---|
| 121 | + if (!power_key_dev) |
|---|
| 122 | + return -ENODEV; |
|---|
| 123 | + |
|---|
| 124 | + input_report_key(power_key_dev, KEY_POWER, 1); |
|---|
| 125 | + input_sync(power_key_dev); |
|---|
| 126 | + msleep(20); |
|---|
| 127 | + input_report_key(power_key_dev, KEY_POWER, 0); |
|---|
| 128 | + input_sync(power_key_dev); |
|---|
| 129 | + |
|---|
| 130 | + return 0; |
|---|
| 131 | +} |
|---|
| 132 | + |
|---|
| 116 | 133 | static irqreturn_t rfkill_rk_wake_host_irq(int irq, void *dev) |
|---|
| 117 | 134 | { |
|---|
| 118 | 135 | struct rfkill_rk_data *rfkill = dev; |
|---|
| .. | .. |
|---|
| 123 | 140 | |
|---|
| 124 | 141 | wake_lock_timeout(&rfkill->bt_irq_wl, |
|---|
| 125 | 142 | msecs_to_jiffies(BT_IRQ_WAKELOCK_TIMEOUT)); |
|---|
| 143 | + |
|---|
| 144 | + if (rfkill->enable_power_key) |
|---|
| 145 | + return IRQ_WAKE_THREAD; |
|---|
| 146 | + |
|---|
| 147 | + return IRQ_HANDLED; |
|---|
| 148 | +} |
|---|
| 149 | + |
|---|
| 150 | +static irqreturn_t rfkill_rk_wake_host_irq_thread(int irq, void *dev) |
|---|
| 151 | +{ |
|---|
| 152 | + rfkill_rk_power_key_up(); |
|---|
| 126 | 153 | |
|---|
| 127 | 154 | return IRQ_HANDLED; |
|---|
| 128 | 155 | } |
|---|
| .. | .. |
|---|
| 165 | 192 | LOG("Request irq for bt wakeup host\n"); |
|---|
| 166 | 193 | irq->irq = gpio_to_irq(irq->gpio.io); |
|---|
| 167 | 194 | sprintf(irq->name, "%s_irq", irq->gpio.name); |
|---|
| 168 | | - ret = request_irq(irq->irq, rfkill_rk_wake_host_irq, |
|---|
| 169 | | - (irq->gpio.enable == GPIO_ACTIVE_LOW) ? |
|---|
| 170 | | - IRQF_TRIGGER_FALLING : |
|---|
| 171 | | - IRQF_TRIGGER_RISING, |
|---|
| 172 | | - irq->name, rfkill); |
|---|
| 195 | + ret = request_threaded_irq(irq->irq, rfkill_rk_wake_host_irq, |
|---|
| 196 | + rfkill_rk_wake_host_irq_thread, |
|---|
| 197 | + IRQF_ONESHOT | ((irq->gpio.enable == GPIO_ACTIVE_LOW) ? |
|---|
| 198 | + IRQF_TRIGGER_FALLING : |
|---|
| 199 | + IRQF_TRIGGER_RISING), |
|---|
| 200 | + irq->name, rfkill); |
|---|
| 173 | 201 | if (ret) |
|---|
| 174 | 202 | goto fail2; |
|---|
| 175 | 203 | rfkill->irq_req = 1; |
|---|
| .. | .. |
|---|
| 271 | 299 | |
|---|
| 272 | 300 | return 0; |
|---|
| 273 | 301 | } |
|---|
| 274 | | -static int proc_rk_set_power(void *data, bool blocked) |
|---|
| 302 | + |
|---|
| 303 | +static int rfkill_rk_set_power(void *data, bool blocked) |
|---|
| 275 | 304 | { |
|---|
| 276 | 305 | struct rfkill_rk_data *rfkill = data; |
|---|
| 277 | 306 | struct rfkill_rk_gpio *wake_host = &rfkill->pdata->wake_host_irq.gpio; |
|---|
| .. | .. |
|---|
| 363 | 392 | LOG("%s: cannot get wifi power state!\n", __func__); |
|---|
| 364 | 393 | return -EPERM; |
|---|
| 365 | 394 | } |
|---|
| 366 | | - if (!wifi_power) { |
|---|
| 367 | | - LOG("%s: bt will set vbat to low\n", __func__); |
|---|
| 368 | | - rfkill_set_wifi_bt_power(0); |
|---|
| 369 | | - } else { |
|---|
| 370 | | - LOG("%s: bt shouldn't control the vbat\n", __func__); |
|---|
| 371 | | - } |
|---|
| 372 | | - } |
|---|
| 373 | | - } |
|---|
| 374 | | - |
|---|
| 375 | | - return 0; |
|---|
| 376 | | -} |
|---|
| 377 | | - |
|---|
| 378 | | - |
|---|
| 379 | | -static int rfkill_rk_set_power(void *data, bool blocked) |
|---|
| 380 | | -{ |
|---|
| 381 | | - struct rfkill_rk_data *rfkill = data; |
|---|
| 382 | | - struct rfkill_rk_gpio *wake_host = &rfkill->pdata->wake_host_irq.gpio; |
|---|
| 383 | | - struct rfkill_rk_gpio *poweron = &rfkill->pdata->poweron_gpio; |
|---|
| 384 | | - struct rfkill_rk_gpio *reset = &rfkill->pdata->reset_gpio; |
|---|
| 385 | | - struct rfkill_rk_gpio *rts = &rfkill->pdata->rts_gpio; |
|---|
| 386 | | - struct pinctrl *pinctrl = rfkill->pdata->pinctrl; |
|---|
| 387 | | - int wifi_power = 0; |
|---|
| 388 | | - bool toggle = false; |
|---|
| 389 | | - |
|---|
| 390 | | - DBG("Enter %s\n", __func__); |
|---|
| 391 | | - |
|---|
| 392 | | - DBG("Set blocked:%d\n", blocked); |
|---|
| 393 | | - |
|---|
| 394 | | - toggle = rfkill->pdata->power_toggle; |
|---|
| 395 | | - |
|---|
| 396 | | - if (toggle) { |
|---|
| 397 | | - if (rfkill_get_wifi_power_state(&wifi_power)) { |
|---|
| 398 | | - LOG("%s: cannot get wifi power state!\n", __func__); |
|---|
| 399 | | - return -1; |
|---|
| 400 | | - } |
|---|
| 401 | | - } |
|---|
| 402 | | - |
|---|
| 403 | | - DBG("%s: toggle = %s\n", __func__, toggle ? "true" : "false"); |
|---|
| 404 | | - |
|---|
| 405 | | - if (!blocked) { |
|---|
| 406 | | - if (toggle) { |
|---|
| 407 | | - rfkill_set_wifi_bt_power(1); |
|---|
| 408 | | - msleep(100); |
|---|
| 409 | | - } |
|---|
| 410 | | - |
|---|
| 411 | | - rfkill_rk_sleep_bt(BT_WAKEUP); // ensure bt is wakeup |
|---|
| 412 | | - |
|---|
| 413 | | - if (gpio_is_valid(wake_host->io)) { |
|---|
| 414 | | - LOG("%s: set bt wake_host high!\n", __func__); |
|---|
| 415 | | - gpio_direction_output(wake_host->io, 1); |
|---|
| 416 | | - msleep(20); |
|---|
| 417 | | - } |
|---|
| 418 | | - |
|---|
| 419 | | - if (gpio_is_valid(poweron->io)) { |
|---|
| 420 | | - if (gpio_get_value(poweron->io) == !poweron->enable) { |
|---|
| 421 | | - gpio_direction_output(poweron->io, |
|---|
| 422 | | - !poweron->enable); |
|---|
| 423 | | - msleep(20); |
|---|
| 424 | | - gpio_direction_output(poweron->io, |
|---|
| 425 | | - poweron->enable); |
|---|
| 426 | | - msleep(20); |
|---|
| 427 | | - if (gpio_is_valid(wake_host->io)) |
|---|
| 428 | | - gpio_direction_input(wake_host->io); |
|---|
| 429 | | - } |
|---|
| 430 | | - } |
|---|
| 431 | | - |
|---|
| 432 | | - if (gpio_is_valid(reset->io)) { |
|---|
| 433 | | - if (gpio_get_value(reset->io) == !reset->enable) { |
|---|
| 434 | | - gpio_direction_output(reset->io, |
|---|
| 435 | | - !reset->enable); |
|---|
| 436 | | - msleep(20); |
|---|
| 437 | | - gpio_direction_output(reset->io, reset->enable); |
|---|
| 438 | | - } |
|---|
| 439 | | - } |
|---|
| 440 | | - |
|---|
| 441 | | - if (pinctrl && gpio_is_valid(rts->io)) { |
|---|
| 442 | | - pinctrl_select_state(pinctrl, rts->gpio_state); |
|---|
| 443 | | - LOG("ENABLE UART_RTS\n"); |
|---|
| 444 | | - gpio_direction_output(rts->io, rts->enable); |
|---|
| 445 | | - msleep(100); |
|---|
| 446 | | - LOG("DISABLE UART_RTS\n"); |
|---|
| 447 | | - gpio_direction_output(rts->io, !rts->enable); |
|---|
| 448 | | - pinctrl_select_state(pinctrl, rts->default_state); |
|---|
| 449 | | - } |
|---|
| 450 | | - |
|---|
| 451 | | - bt_power_state = 1; |
|---|
| 452 | | - LOG("bt turn on power\n"); |
|---|
| 453 | | - rfkill_rk_setup_wake_irq(rfkill, 1); |
|---|
| 454 | | - } else { |
|---|
| 455 | | - if (gpio_is_valid(poweron->io)) { |
|---|
| 456 | | - if (gpio_get_value(poweron->io) == poweron->enable) { |
|---|
| 457 | | - gpio_direction_output(poweron->io, |
|---|
| 458 | | - poweron->enable); |
|---|
| 459 | | - msleep(20); |
|---|
| 460 | | - } |
|---|
| 461 | | - } |
|---|
| 462 | | - |
|---|
| 463 | | - bt_power_state = 0; |
|---|
| 464 | | - LOG("bt shut off power\n"); |
|---|
| 465 | | - if (gpio_is_valid(reset->io)) { |
|---|
| 466 | | - if (gpio_get_value(reset->io) == reset->enable) { |
|---|
| 467 | | - gpio_direction_output(reset->io, |
|---|
| 468 | | - reset->enable); |
|---|
| 469 | | - msleep(20); |
|---|
| 470 | | - } |
|---|
| 471 | | - } |
|---|
| 472 | | - if (toggle) { |
|---|
| 473 | 395 | if (!wifi_power) { |
|---|
| 474 | 396 | LOG("%s: bt will set vbat to low\n", __func__); |
|---|
| 475 | 397 | rfkill_set_wifi_bt_power(0); |
|---|
| .. | .. |
|---|
| 592 | 514 | return -EFAULT; |
|---|
| 593 | 515 | |
|---|
| 594 | 516 | DBG("btwrite %c\n", b); |
|---|
| 595 | | - /* HCI_DEV_WRITE */ |
|---|
| 596 | 517 | if (b != '0') |
|---|
| 597 | | - proc_rk_set_power(g_rfkill, 0); |
|---|
| 518 | + rfkill_rk_sleep_bt(BT_WAKEUP); |
|---|
| 598 | 519 | else |
|---|
| 599 | | - proc_rk_set_power(g_rfkill, 1); |
|---|
| 520 | + rfkill_rk_sleep_bt(BT_SLEEP); |
|---|
| 521 | + |
|---|
| 522 | + return count; |
|---|
| 523 | +} |
|---|
| 524 | + |
|---|
| 525 | +static ssize_t bluesleep_read_proc_powerupkey(struct file *file, |
|---|
| 526 | + char __user *buffer, size_t count, |
|---|
| 527 | + loff_t *data) |
|---|
| 528 | +{ |
|---|
| 529 | + struct rfkill_rk_data *rfkill = g_rfkill; |
|---|
| 530 | + char src[2]; |
|---|
| 531 | + |
|---|
| 532 | + if (*data >= 1) |
|---|
| 533 | + return 0; |
|---|
| 534 | + |
|---|
| 535 | + if (!rfkill) |
|---|
| 536 | + return -EFAULT; |
|---|
| 537 | + |
|---|
| 538 | + src[0] = rfkill->enable_power_key ? '1' : '0'; |
|---|
| 539 | + src[1] = '\n'; |
|---|
| 540 | + if (copy_to_user(buffer, src, 2)) |
|---|
| 541 | + return -EFAULT; |
|---|
| 542 | + *data = 1; |
|---|
| 543 | + |
|---|
| 544 | + return 2; |
|---|
| 545 | +} |
|---|
| 546 | + |
|---|
| 547 | +static ssize_t bluesleep_write_proc_powerupkey(struct file *file, |
|---|
| 548 | + const char __user *buffer, |
|---|
| 549 | + size_t count, loff_t *data) |
|---|
| 550 | +{ |
|---|
| 551 | + char b; |
|---|
| 552 | + struct rfkill_rk_data *rfkill = g_rfkill; |
|---|
| 553 | + |
|---|
| 554 | + if (!rfkill) |
|---|
| 555 | + return -EFAULT; |
|---|
| 556 | + |
|---|
| 557 | + if (count < 1) |
|---|
| 558 | + return -EINVAL; |
|---|
| 559 | + |
|---|
| 560 | + if (copy_from_user(&b, buffer, 1)) |
|---|
| 561 | + return -EFAULT; |
|---|
| 562 | + |
|---|
| 563 | + if (b != '0') |
|---|
| 564 | + rfkill->enable_power_key = true; |
|---|
| 565 | + else |
|---|
| 566 | + rfkill->enable_power_key = false; |
|---|
| 600 | 567 | |
|---|
| 601 | 568 | return count; |
|---|
| 602 | 569 | } |
|---|
| .. | .. |
|---|
| 700 | 667 | .write = bluesleep_write_proc_btwrite, |
|---|
| 701 | 668 | }; |
|---|
| 702 | 669 | |
|---|
| 670 | +static const struct file_operations bluesleep_powerupkey = { |
|---|
| 671 | + .owner = THIS_MODULE, |
|---|
| 672 | + .read = bluesleep_read_proc_powerupkey, |
|---|
| 673 | + .write = bluesleep_write_proc_powerupkey, |
|---|
| 674 | +}; |
|---|
| 675 | + |
|---|
| 676 | +static int rfkill_rk_register_power_key(void) |
|---|
| 677 | +{ |
|---|
| 678 | + int ret = 0; |
|---|
| 679 | + |
|---|
| 680 | + /* register input device */ |
|---|
| 681 | + power_key_dev = input_allocate_device(); |
|---|
| 682 | + if (!power_key_dev) { |
|---|
| 683 | + LOG("ir_dev: not enough memory for input device\n"); |
|---|
| 684 | + return -ENOMEM; |
|---|
| 685 | + } |
|---|
| 686 | + |
|---|
| 687 | + power_key_dev->name = "bt-powerkey"; |
|---|
| 688 | + power_key_dev->id.bustype = BUS_HOST; |
|---|
| 689 | + |
|---|
| 690 | + power_key_dev->evbit[0] = BIT_MASK(EV_KEY); |
|---|
| 691 | + set_bit(KEY_POWER, power_key_dev->keybit); |
|---|
| 692 | + |
|---|
| 693 | + ret = input_register_device(power_key_dev); |
|---|
| 694 | + if (ret) { |
|---|
| 695 | + input_free_device(power_key_dev); |
|---|
| 696 | + LOG("ir_rx_init: register input device exception, exit\n"); |
|---|
| 697 | + return -EBUSY; |
|---|
| 698 | + } |
|---|
| 699 | + |
|---|
| 700 | + return ret; |
|---|
| 701 | +} |
|---|
| 702 | + |
|---|
| 703 | 703 | static int rfkill_rk_probe(struct platform_device *pdev) |
|---|
| 704 | 704 | { |
|---|
| 705 | 705 | struct rfkill_rk_data *rfkill; |
|---|
| .. | .. |
|---|
| 766 | 766 | goto fail_alloc; |
|---|
| 767 | 767 | } |
|---|
| 768 | 768 | |
|---|
| 769 | + /* read/write proc entries */ |
|---|
| 770 | + ent = proc_create("powerupkey", 0, sleep_dir, &bluesleep_powerupkey); |
|---|
| 771 | + if (!ent) { |
|---|
| 772 | + LOG("Unable to create /proc/%s/powerupkey entry", PROC_DIR); |
|---|
| 773 | + ret = -ENOMEM; |
|---|
| 774 | + goto fail_alloc; |
|---|
| 775 | + } |
|---|
| 776 | + |
|---|
| 769 | 777 | DBG("init gpio\n"); |
|---|
| 770 | 778 | |
|---|
| 771 | 779 | ret = rfkill_rk_setup_gpio(pdev, &pdata->poweron_gpio, pdata->name, |
|---|
| .. | .. |
|---|
| 824 | 832 | |
|---|
| 825 | 833 | LOG("%s device registered.\n", pdata->name); |
|---|
| 826 | 834 | |
|---|
| 835 | + if (rfkill_rk_register_power_key() != 0) |
|---|
| 836 | + goto fail_rfkill; |
|---|
| 837 | + |
|---|
| 827 | 838 | return 0; |
|---|
| 828 | 839 | |
|---|
| 829 | 840 | fail_rfkill: |
|---|
| 830 | 841 | rfkill_destroy(rfkill->rfkill_dev); |
|---|
| 831 | 842 | fail_alloc: |
|---|
| 832 | 843 | |
|---|
| 844 | + remove_proc_entry("powerupkey", sleep_dir); |
|---|
| 833 | 845 | remove_proc_entry("btwrite", sleep_dir); |
|---|
| 834 | 846 | remove_proc_entry("lpm", sleep_dir); |
|---|
| 835 | 847 | fail_setup_wake_irq: |
|---|
| .. | .. |
|---|
| 848 | 860 | |
|---|
| 849 | 861 | rfkill_unregister(rfkill->rfkill_dev); |
|---|
| 850 | 862 | rfkill_destroy(rfkill->rfkill_dev); |
|---|
| 863 | + remove_proc_entry("powerupkey", sleep_dir); |
|---|
| 864 | + remove_proc_entry("btwrite", sleep_dir); |
|---|
| 865 | + remove_proc_entry("lpm", sleep_dir); |
|---|
| 851 | 866 | |
|---|
| 867 | + input_unregister_device(power_key_dev); |
|---|
| 852 | 868 | cancel_delayed_work_sync(&rfkill->bt_sleep_delay_work); |
|---|
| 853 | 869 | |
|---|
| 854 | 870 | // free gpio |
|---|