Hi Marcel, On Tue, Aug 21, 2018 at 7:43 PM, Marcel Holtmann <marcel@xxxxxxxxxxxx> wrote: > Hi Raghuram, > >> Driver to add INTL6205 platform ACPI device to enable out >> of band GPIO signalling to handle rfkill and reset the >> bluetooth controller. Expose an interface in kernel to >> assert GPIO signal. >> >> Signed-off-by: Chethan T N <chethan.tumkur.narayan@xxxxxxxxx> >> Signed-off-by: Sukumar Ghorai <sukumar.ghorai@xxxxxxxxx> >> Signed-off-by: Raghuram Hegde <raghuram.hegde@xxxxxxxxx> >> --- >> drivers/bluetooth/btintel.c | 68 +++++++++++++++++++++++++++++++++++++++++++++ >> drivers/bluetooth/btintel.h | 6 ++++ >> 2 files changed, 74 insertions(+) >> >> diff --git a/drivers/bluetooth/btintel.c b/drivers/bluetooth/btintel.c >> index 5270d5513201..538cd6b6c524 100644 >> --- a/drivers/bluetooth/btintel.c >> +++ b/drivers/bluetooth/btintel.c >> @@ -24,6 +24,9 @@ >> #include <linux/module.h> >> #include <linux/firmware.h> >> #include <linux/regmap.h> >> +#include <linux/acpi.h> >> +#include <linux/platform_device.h> >> +#include <linux/gpio/consumer.h> >> #include <asm/unaligned.h> >> >> #include <net/bluetooth/bluetooth.h> >> @@ -375,6 +378,71 @@ int btintel_read_version(struct hci_dev *hdev, struct intel_version *ver) >> } >> EXPORT_SYMBOL_GPL(btintel_read_version); >> >> +static struct gpio_desc *reset_gpio_handler; > > this thing is so inherent broken. Please never ever do that. If there are two reset handlers or two Intel USB devices connected things will go horrible wrong. Is it fine to keep list of the descriptors handlers for all the Intel devices connected? > >> +void btintel_reset_bt(struct hci_dev *hdev, unsigned char code) >> +{ >> + if (!reset_gpio_handler) >> + return; >> + >> + gpiod_set_value(reset_gpio_handler, 0); >> + mdelay(100); >> + gpiod_set_value(reset_gpio_handler, 1); >> +} >> +EXPORT_SYMBOL_GPL(btintel_reset_bt); > > What happens here when you do this? Does the Bluetooth USB device disconnect from the USB bus and gets re-enumerated and the btusb_probe() routine gets called again? Yes, USB re-enumeration will be done and btusb_proble() function will be called. > > If this is the case, then I have no idea how many times I have to explain. It is a platfrom reset switch and using RFKILL for this is acceptable. Thought of make this platform independent by exposing ACPI object "INTL6205", by doing so in case different platforms if the ACPI object is exposed by the platform then this driver would be loaded. Kindly please do suggest. > >> + >> +static const struct acpi_gpio_params reset_gpios = { 0, 0, false }; >> +static const struct acpi_gpio_mapping acpi_btintel_gpios[] = { >> + { "reset-gpios", &reset_gpios, 1 }, >> + { }, >> +}; >> + >> +static int btintel_probe(struct platform_device *pdev) >> +{ >> + struct device *hdev; >> + >> + hdev = devm_kzalloc(&pdev->dev, sizeof(*hdev), GFP_KERNEL); >> + if (!hdev) >> + return -ENOMEM; > > What is this doing. This makes no sense. > >> + >> + if (!ACPI_HANDLE(&pdev->dev)) >> + return -ENODEV; >> + >> + if (acpi_dev_add_driver_gpios(ACPI_COMPANION(&pdev->dev), >> + acpi_btintel_gpios)) >> + return -EINVAL; >> + >> + reset_gpio_handler = devm_gpiod_get_optional(&pdev->dev, >> + "reset", GPIOD_OUT_HIGH); >> + if (IS_ERR(reset_gpio_handler)) >> + return PTR_ERR(reset_gpio_handler); >> + >> + return 0; >> +} >> + >> +static int btintel_remove(struct platform_device *pdev) >> +{ >> + acpi_dev_remove_driver_gpios(ACPI_COMPANION(&pdev->dev)); >> + return 0; >> +} >> + >> +#ifdef CONFIG_ACPI >> +static const struct acpi_device_id btintel_acpi_match[] = { >> + { "INTL6205", 0 }, >> + { }, >> +}; >> +MODULE_DEVICE_TABLE(acpi, btintel_acpi_match); >> +#endif >> + >> +static struct platform_driver btintel_driver = { >> + .probe = btintel_probe, >> + .remove = btintel_remove, >> + .driver = { >> + .name = "btintel", >> + .acpi_match_table = ACPI_PTR(btintel_acpi_match), >> + }, >> +}; >> +module_platform_driver(btintel_driver); > > I am not convinced this actually works since now we have two module_init and module_exit functions. We have tested these changes on two different platform and it is working. > >> + >> /* ------- REGMAP IBT SUPPORT ------- */ >> >> #define IBT_REG_MODE_8BIT 0x00 >> diff --git a/drivers/bluetooth/btintel.h b/drivers/bluetooth/btintel.h >> index 41c642cc523f..7686fab52054 100644 >> --- a/drivers/bluetooth/btintel.h >> +++ b/drivers/bluetooth/btintel.h >> @@ -94,6 +94,7 @@ int btintel_load_ddc_config(struct hci_dev *hdev, const char *ddc_name); >> int btintel_set_event_mask(struct hci_dev *hdev, bool debug); >> int btintel_set_event_mask_mfg(struct hci_dev *hdev, bool debug); >> int btintel_read_version(struct hci_dev *hdev, struct intel_version *ver); >> +void btintel_reset_bt(struct hci_dev *hdev, unsigned char code); >> >> struct regmap *btintel_regmap_init(struct hci_dev *hdev, u16 opcode_read, >> u16 opcode_write); >> @@ -171,6 +172,11 @@ static inline int btintel_read_version(struct hci_dev *hdev, >> return -EOPNOTSUPP; >> } >> >> +static inline void btintel_reset_bt(struct hci_dev *hdev, unsigned char code) >> +{ >> + return -EOPNOTSUPP; >> +} >> + >> static inline struct regmap *btintel_regmap_init(struct hci_dev *hdev, >> u16 opcode_read, >> u16 opcode_write) > > Regards > > Marcel > Thanks and Regards Chethan