Retrieve the Interruption used by BCM device, which can be declared as Interruption or GpioInt in the ACPI table. Configure BCM device to wake-up the host. Enable IRQ wake while suspended. host_wake_active parameter of Vendor Specific Command should not be configured with the same value for BCM2E39 and BCM2E67. So, retrieve the irq polarity used from the ACPI table. Unfortunately, ACPI table for BCM2E39 is not correct. So, add fix_acpi_irq parameter to bcm_device to be able to revert host_wake_active when sending sleep parameters. Signed-off-by: Frederic Danis <frederic.danis@xxxxxxxxxxxxxxx> --- drivers/bluetooth/hci_bcm.c | 141 ++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 136 insertions(+), 5 deletions(-) diff --git a/drivers/bluetooth/hci_bcm.c b/drivers/bluetooth/hci_bcm.c index 835bfab..1440a56 100644 --- a/drivers/bluetooth/hci_bcm.c +++ b/drivers/bluetooth/hci_bcm.c @@ -31,6 +31,7 @@ #include <linux/clk.h> #include <linux/gpio/consumer.h> #include <linux/tty.h> +#include <linux/interrupt.h> #include <net/bluetooth/bluetooth.h> #include <net/bluetooth/hci_core.h> @@ -38,10 +39,14 @@ #include "btbcm.h" #include "hci_uart.h" +#define BCM_NO_FIX 0 +#define BCM_FIX_ACPI_IRQ 1 + struct bcm_device { struct list_head list; struct platform_device *pdev; + bool fix_acpi_irq; const char *name; struct gpio_desc *device_wakeup; @@ -55,6 +60,8 @@ struct bcm_device { #ifdef CONFIG_PM_SLEEP struct hci_uart *hu; bool is_suspended; /* suspend/resume flag */ + int irq; + u8 irq_polarity; #endif }; @@ -149,6 +156,46 @@ static int bcm_gpio_set_power(struct bcm_device *dev, bool powered) return 0; } +static const struct bcm_set_sleep_mode default_sleep_params = { + .sleep_mode = 1, /* 0=Disabled, 1=UART, 2=Reserved, 3=USB */ + .idle_host = 2, /* idle threshold HOST, in 300ms */ + .idle_dev = 2, /* idle threshold device, in 300ms */ + .bt_wake_active = 1, /* BT_WAKE active mode: 1 = high, 0 = low */ + .host_wake_active = 0, /* HOST_WAKE active mode: 1 = high, 0 = low */ + .allow_host_sleep = 1, /* Allow host sleep in SCO flag */ + .combine_modes = 0, /* Combine sleep and LPM flag */ + .tristate_control = 0, /* Allow tri-state control of UART tx flag */ + /* Irrelevant USB flags */ + .usb_auto_sleep = 0, + .usb_resume_timeout = 0, + .pulsed_host_wake = 0, + .break_to_host = 0 +}; + +static int bcm_setup_sleep(struct hci_uart *hu) +{ + struct bcm_data *bcm = hu->priv; + struct sk_buff *skb; + struct bcm_set_sleep_mode sleep_params = default_sleep_params; + + sleep_params.host_wake_active = !bcm->dev->irq_polarity; + if (bcm->dev->fix_acpi_irq) + sleep_params.host_wake_active = !sleep_params.host_wake_active; + + skb = __hci_cmd_sync(hu->hdev, 0xfc27, sizeof(sleep_params), + &sleep_params, HCI_INIT_TIMEOUT); + if (IS_ERR(skb)) { + int err = PTR_ERR(skb); + BT_ERR("Sleep VSC failed (%d)", err); + return err; + } + kfree_skb(skb); + + BT_DBG("bcm_setup Set Sleep Parameters VSC succeeded"); + + return 0; +} + static int bcm_open(struct hci_uart *hu) { struct bcm_data *bcm; @@ -193,15 +240,20 @@ static int bcm_open(struct hci_uart *hu) static int bcm_close(struct hci_uart *hu) { struct bcm_data *bcm = hu->priv; + struct bcm_device *bdev = bcm->dev; BT_DBG("hu %p", hu); /* Protect bcm->dev against removal of the device or driver */ spin_lock(&bcm_device_lock); - if (bcm_device_exists(bcm->dev)) { - bcm_gpio_set_power(bcm->dev, false); + if (bcm_device_exists(bdev)) { + bcm_gpio_set_power(bdev, false); #ifdef CONFIG_PM_SLEEP - bcm->dev->hu = NULL; + bdev->hu = NULL; + if (device_can_wakeup(&bdev->pdev->dev)) { + devm_free_irq(&bdev->pdev->dev, bdev->irq, bdev); + device_init_wakeup(&bdev->pdev->dev, false); + } #endif } spin_unlock(&bcm_device_lock); @@ -225,8 +277,21 @@ static int bcm_flush(struct hci_uart *hu) return 0; } +static irqreturn_t bcm_host_wake(int irq, void *data) +{ + struct bcm_device *dev = data; + + BT_DBG("Host wake IRQ for %s", dev->name); + + return IRQ_HANDLED; +} + static int bcm_setup(struct hci_uart *hu) { +#ifdef CONFIG_PM_SLEEP + struct bcm_data *bcm = hu->priv; + struct bcm_device *bdev = bcm->dev; +#endif char fw_name[64]; const struct firmware *fw; unsigned int speed; @@ -281,6 +346,30 @@ finalize: release_firmware(fw); err = btbcm_finalize(hu->hdev); +#ifdef CONFIG_PM_SLEEP + if (err) + return err; + + /* If it is not a platform device, do not enable PM functionalities */ + spin_lock(&bcm_device_lock); + if (!bcm_device_exists(bdev)) + goto unlock; + + if (bdev->irq > 0) { + err = devm_request_irq(&bdev->pdev->dev, bdev->irq, + bcm_host_wake, IRQF_TRIGGER_RISING, + "host_wake", bdev); + if (err) + goto unlock; + + device_init_wakeup(&bdev->pdev->dev, true); + } + +unlock: + spin_unlock(&bcm_device_lock); + + err = bcm_setup_sleep(hu); +#endif return err; } @@ -335,6 +424,7 @@ static struct sk_buff *bcm_dequeue(struct hci_uart *hu) static int bcm_suspend(struct device *dev) { struct bcm_device *bdev = platform_get_drvdata(to_platform_device(dev)); + int error; BT_DBG("suspend (%p): is_suspended %d", bdev, bdev->is_suspended); @@ -357,6 +447,12 @@ static int bcm_suspend(struct device *dev) mdelay(15); } + if (device_may_wakeup(&bdev->pdev->dev)) { + error = enable_irq_wake(bdev->irq); + if (!error) + BT_DBG("BCM irq: enabled"); + } + unlock: spin_unlock(&bcm_device_lock); @@ -375,6 +471,11 @@ static int bcm_resume(struct device *dev) if (!bdev->hu) goto unlock; + if (device_may_wakeup(&bdev->pdev->dev)) { + disable_irq_wake(bdev->irq); + BT_DBG("BCM irq: disabled"); + } + if (bdev->device_wakeup) { gpiod_set_value(bdev->device_wakeup, true); BT_DBG("resume, delaying 15 ms"); @@ -397,10 +498,12 @@ unlock: static const struct acpi_gpio_params device_wakeup_gpios = { 0, 0, false }; static const struct acpi_gpio_params shutdown_gpios = { 1, 0, false }; +static const struct acpi_gpio_params host_wakeup_gpios = { 2, 0, false }; static const struct acpi_gpio_mapping acpi_bcm_default_gpios[] = { { "device-wakeup-gpios", &device_wakeup_gpios, 1 }, { "shutdown-gpios", &shutdown_gpios, 1 }, + { "host-wakeup-gpios", &host_wakeup_gpios, 1 }, { }, }; @@ -415,6 +518,17 @@ static int bcm_resource(struct acpi_resource *ares, void *data) sb = &ares->data.uart_serial_bus; if (sb->type == ACPI_RESOURCE_SERIAL_TYPE_UART) dev->init_speed = sb->default_baud_rate; + } else if (ares->type == ACPI_RESOURCE_TYPE_EXTENDED_IRQ) { + struct acpi_resource_extended_irq *irq; + + irq = &ares->data.extended_irq; + dev->irq_polarity = irq->polarity; + } else if (ares->type == ACPI_RESOURCE_TYPE_GPIO) { + struct acpi_resource_gpio *gpio; + + gpio = &ares->data.gpio; + if (gpio->connection_type == ACPI_RESOURCE_GPIO_TYPE_INT) + dev->irq_polarity = gpio->polarity; } /* Always tell the ACPI core to skip this resource */ @@ -433,6 +547,8 @@ static int bcm_acpi_probe(struct bcm_device *dev) if (!id) return -ENODEV; + dev->fix_acpi_irq = !!id->driver_data; + /* Retrieve GPIO data */ dev->name = dev_name(&pdev->dev); ret = acpi_dev_add_driver_gpios(ACPI_COMPANION(&pdev->dev), @@ -453,6 +569,21 @@ static int bcm_acpi_probe(struct bcm_device *dev) if (IS_ERR(dev->shutdown)) return PTR_ERR(dev->shutdown); + /* IRQ can be declared in ACPI table as Interrupt or GpioInt */ + dev->irq = platform_get_irq(pdev, 0); + if (dev->irq <= 0) { + struct gpio_desc *gpio; + + gpio = devm_gpiod_get_optional(&pdev->dev, "host-wakeup", + GPIOD_IN); + if (IS_ERR(gpio)) + return PTR_ERR(gpio); + + dev->irq = gpiod_to_irq(gpio); + } + + dev_info(&pdev->dev, "BCM irq: %d\n", dev->irq); + /* Make sure at-least one of the GPIO is defined and that * a name is specified for this instance */ @@ -545,8 +676,8 @@ static const struct hci_uart_proto bcm_proto = { #ifdef CONFIG_ACPI static const struct acpi_device_id bcm_acpi_match[] = { - { "BCM2E39", 0 }, - { "BCM2E67", 0 }, + { "BCM2E39", BCM_FIX_ACPI_IRQ }, + { "BCM2E67", BCM_NO_FIX }, { }, }; MODULE_DEVICE_TABLE(acpi, bcm_acpi_match); -- 1.9.1 -- To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html