All of lore.kernel.org
 help / color / mirror / Atom feed
* [PATCH v2] bluetooth: core: RfC - add basic LED triggers
@ 2016-01-01 16:23 Heiner Kallweit
  2016-01-04 18:19 ` Marcel Holtmann
  0 siblings, 1 reply; 7+ messages in thread
From: Heiner Kallweit @ 2016-01-01 16:23 UTC (permalink / raw)
  To: Marcel Holtmann; +Cc: Johan Hedberg, linux-bluetooth, Johannes Berg

In the bluetooth subsystem I miss the option to use LEDs to indicate
the status. This patch is based on the LED triggers in mac80211 and
adds two basic LED triggers (powered up, device connected) plus
a config option to enable the BT LED triggers.

It works fine on my system however I don't know the bluetooth stack
in too much detail and therefore appreciate any comment.

Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
---
v2:
- completely rewrote the code to be as generic as possible and minimize
  the changes to struct hci_dev
- make the details of the LED triggers opaque to struct hci_dev
- avoid using ifdef CONFIG_BT_LEDS in hci_core.h
- start with the radio trigger only
- factor out the generic basic led trigger code to facilitate adding
  further basic LED triggers
- fix indentation in Kconfig
- rename led.[ch] to leds.[ch]
---
 include/net/bluetooth/hci_core.h |  3 ++
 net/bluetooth/Kconfig            |  9 ++++
 net/bluetooth/Makefile           |  1 +
 net/bluetooth/hci_core.c         |  8 ++++
 net/bluetooth/leds.c             | 99 ++++++++++++++++++++++++++++++++++++++++
 net/bluetooth/leds.h             | 17 +++++++
 6 files changed, 137 insertions(+)
 create mode 100644 net/bluetooth/leds.c
 create mode 100644 net/bluetooth/leds.h

diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h
index c95e032..4ffec8b 100644
--- a/include/net/bluetooth/hci_core.h
+++ b/include/net/bluetooth/hci_core.h
@@ -25,6 +25,7 @@
 #ifndef __HCI_CORE_H
 #define __HCI_CORE_H
 
+#include <linux/leds.h>
 #include <net/bluetooth/hci.h>
 #include <net/bluetooth/hci_sock.h>
 
@@ -395,6 +396,8 @@ struct hci_dev {
 	struct delayed_work	rpa_expired;
 	bdaddr_t		rpa;
 
+	struct led_trigger *radio_led;
+
 	int (*open)(struct hci_dev *hdev);
 	int (*close)(struct hci_dev *hdev);
 	int (*flush)(struct hci_dev *hdev);
diff --git a/net/bluetooth/Kconfig b/net/bluetooth/Kconfig
index 95d1a66..06c31b9 100644
--- a/net/bluetooth/Kconfig
+++ b/net/bluetooth/Kconfig
@@ -69,6 +69,15 @@ config BT_6LOWPAN
 	help
 	  IPv6 compression over Bluetooth Low Energy.
 
+config BT_LEDS
+	bool "Enable LED triggers"
+	depends on BT
+	depends on LEDS_CLASS
+	select LEDS_TRIGGERS
+	help
+	  This option selects a few LED triggers for different
+	  Bluetooth events.
+
 config BT_SELFTEST
 	bool "Bluetooth self testing support"
 	depends on BT && DEBUG_KERNEL
diff --git a/net/bluetooth/Makefile b/net/bluetooth/Makefile
index 2b15ae8..b3ff12e 100644
--- a/net/bluetooth/Makefile
+++ b/net/bluetooth/Makefile
@@ -17,6 +17,7 @@ bluetooth-y := af_bluetooth.o hci_core.o hci_conn.o hci_event.o mgmt.o \
 
 bluetooth-$(CONFIG_BT_BREDR) += sco.o
 bluetooth-$(CONFIG_BT_HS) += a2mp.o amp.o
+bluetooth-$(CONFIG_BT_LEDS) += leds.o
 bluetooth-$(CONFIG_BT_DEBUGFS) += hci_debugfs.o
 bluetooth-$(CONFIG_BT_SELFTEST) += selftest.o
 
diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c
index 47bcef7..f472b1d 100644
--- a/net/bluetooth/hci_core.c
+++ b/net/bluetooth/hci_core.c
@@ -40,6 +40,7 @@
 #include "hci_request.h"
 #include "hci_debugfs.h"
 #include "smp.h"
+#include "leds.h"
 
 static void hci_rx_work(struct work_struct *work);
 static void hci_cmd_work(struct work_struct *work);
@@ -1395,6 +1396,7 @@ static int hci_dev_do_open(struct hci_dev *hdev)
 		hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
 		set_bit(HCI_UP, &hdev->flags);
 		hci_sock_dev_event(hdev, HCI_DEV_UP);
+		hci_led_radio(hdev, true);
 		if (!hci_dev_test_flag(hdev, HCI_SETUP) &&
 		    !hci_dev_test_flag(hdev, HCI_CONFIG) &&
 		    !hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
@@ -1532,6 +1534,8 @@ int hci_dev_do_close(struct hci_dev *hdev)
 		return 0;
 	}
 
+	hci_led_radio(hdev, false);
+
 	/* Flush RX and TX works */
 	flush_work(&hdev->tx_work);
 	flush_work(&hdev->rx_work);
@@ -3067,6 +3071,8 @@ int hci_register_dev(struct hci_dev *hdev)
 	if (error < 0)
 		goto err_wqueue;
 
+	hci_led_init(hdev);
+
 	hdev->rfkill = rfkill_alloc(hdev->name, &hdev->dev,
 				    RFKILL_TYPE_BLUETOOTH, &hci_rfkill_ops,
 				    hdev);
@@ -3128,6 +3134,8 @@ void hci_unregister_dev(struct hci_dev *hdev)
 
 	id = hdev->id;
 
+	hci_led_exit(hdev);
+
 	write_lock(&hci_dev_list_lock);
 	list_del(&hdev->list);
 	write_unlock(&hci_dev_list_lock);
diff --git a/net/bluetooth/leds.c b/net/bluetooth/leds.c
new file mode 100644
index 0000000..377f643
--- /dev/null
+++ b/net/bluetooth/leds.c
@@ -0,0 +1,99 @@
+/*
+ * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ */
+
+#include <net/bluetooth/bluetooth.h>
+#include <net/bluetooth/hci_core.h>
+
+#include "leds.h"
+
+struct hci_basic_led_trigger {
+	struct led_trigger	led_trigger;
+	atomic_t		cnt;
+};
+
+#define to_hci_basic_led_trigger(arg) container_of(arg, \
+					struct hci_basic_led_trigger, \
+					led_trigger)
+
+static void hci_basic_led(struct led_trigger *trig, bool inc)
+{
+	struct hci_basic_led_trigger *htrig;
+	int cnt;
+
+	if (!trig)
+		return;
+
+	htrig = to_hci_basic_led_trigger(trig);
+
+	if (inc)
+		cnt = atomic_inc_return(&htrig->cnt);
+	else
+		cnt = atomic_dec_return(&htrig->cnt);
+
+	WARN_ON(cnt < 0);
+
+	led_trigger_event(trig, cnt ? LED_FULL : LED_OFF);
+}
+
+void hci_led_radio(struct hci_dev *hdev, bool enabled)
+{
+	hci_basic_led(hdev->radio_led, enabled);
+}
+
+static void hci_basic_led_activate(struct led_classdev *led_cdev)
+{
+	struct hci_basic_led_trigger *htrig;
+	int cnt;
+
+	htrig = to_hci_basic_led_trigger(led_cdev->trigger);
+	cnt = atomic_read(&htrig->cnt);
+
+	led_trigger_event(led_cdev->trigger, cnt ? LED_FULL : LED_OFF);
+}
+
+static struct led_trigger *hci_basic_led_allocate(struct hci_dev *hdev,
+						  const char *name)
+{
+	struct hci_basic_led_trigger *htrig;
+
+	htrig =	devm_kzalloc(&hdev->dev, sizeof(*htrig), GFP_KERNEL);
+	if (!htrig)
+		return NULL;
+
+	atomic_set(&htrig->cnt, 0);
+	htrig->led_trigger.activate = hci_basic_led_activate;
+	htrig->led_trigger.name = devm_kasprintf(&hdev->dev, GFP_KERNEL,
+						 "%s-%s", hdev->name,
+						 name);
+	if (!htrig->led_trigger.name)
+		goto err_alloc;
+
+	if (led_trigger_register(&htrig->led_trigger))
+		goto err_register;
+
+	return &htrig->led_trigger;
+
+err_register:
+	devm_kfree(&hdev->dev, (void *)htrig->led_trigger.name);
+err_alloc:
+	devm_kfree(&hdev->dev, htrig);
+	return NULL;
+}
+
+void hci_led_init(struct hci_dev *hdev)
+{
+	/* initialize radio_led */
+	hdev->radio_led = hci_basic_led_allocate(hdev, "radio");
+}
+
+void hci_led_exit(struct hci_dev *hdev)
+{
+	if (hdev->radio_led)
+		led_trigger_unregister(hdev->radio_led);
+}
+
diff --git a/net/bluetooth/leds.h b/net/bluetooth/leds.h
new file mode 100644
index 0000000..7578cab
--- /dev/null
+++ b/net/bluetooth/leds.h
@@ -0,0 +1,17 @@
+/*
+ * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ */
+
+#ifdef CONFIG_BT_LEDS
+void hci_led_radio(struct hci_dev *hdev, bool enabled);
+void hci_led_init(struct hci_dev *hdev);
+void hci_led_exit(struct hci_dev *hdev);
+#else
+static inline void hci_led_radio(struct hci_dev *hdev, bool enabled) {}
+static inline void hci_led_init(struct hci_dev *hdev) {}
+static inline void hci_led_exit(struct hci_dev *hdev) {}
+#endif
-- 
2.6.4

^ permalink raw reply related	[flat|nested] 7+ messages in thread

* Re: [PATCH v2] bluetooth: core: RfC - add basic LED triggers
  2016-01-01 16:23 [PATCH v2] bluetooth: core: RfC - add basic LED triggers Heiner Kallweit
@ 2016-01-04 18:19 ` Marcel Holtmann
  2016-01-04 21:24   ` Heiner Kallweit
  0 siblings, 1 reply; 7+ messages in thread
From: Marcel Holtmann @ 2016-01-04 18:19 UTC (permalink / raw)
  To: Heiner Kallweit; +Cc: Johan Hedberg, linux-bluetooth, Johannes Berg

Hi Heiner,

> In the bluetooth subsystem I miss the option to use LEDs to indicate
> the status. This patch is based on the LED triggers in mac80211 and
> adds two basic LED triggers (powered up, device connected) plus
> a config option to enable the BT LED triggers.
> 
> It works fine on my system however I don't know the bluetooth stack
> in too much detail and therefore appreciate any comment.
> 
> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
> ---
> v2:
> - completely rewrote the code to be as generic as possible and minimize
>  the changes to struct hci_dev
> - make the details of the LED triggers opaque to struct hci_dev
> - avoid using ifdef CONFIG_BT_LEDS in hci_core.h
> - start with the radio trigger only
> - factor out the generic basic led trigger code to facilitate adding
>  further basic LED triggers
> - fix indentation in Kconfig
> - rename led.[ch] to leds.[ch]
> ---
> include/net/bluetooth/hci_core.h |  3 ++
> net/bluetooth/Kconfig            |  9 ++++
> net/bluetooth/Makefile           |  1 +
> net/bluetooth/hci_core.c         |  8 ++++
> net/bluetooth/leds.c             | 99 ++++++++++++++++++++++++++++++++++++++++
> net/bluetooth/leds.h             | 17 +++++++
> 6 files changed, 137 insertions(+)
> create mode 100644 net/bluetooth/leds.c
> create mode 100644 net/bluetooth/leds.h
> 
> diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h
> index c95e032..4ffec8b 100644
> --- a/include/net/bluetooth/hci_core.h
> +++ b/include/net/bluetooth/hci_core.h
> @@ -25,6 +25,7 @@
> #ifndef __HCI_CORE_H
> #define __HCI_CORE_H
> 
> +#include <linux/leds.h>
> #include <net/bluetooth/hci.h>
> #include <net/bluetooth/hci_sock.h>
> 
> @@ -395,6 +396,8 @@ struct hci_dev {
> 	struct delayed_work	rpa_expired;
> 	bdaddr_t		rpa;
> 
> +	struct led_trigger *radio_led;
> +

I wonder if this one might not be better named power_led instead.

> 	int (*open)(struct hci_dev *hdev);
> 	int (*close)(struct hci_dev *hdev);
> 	int (*flush)(struct hci_dev *hdev);
> diff --git a/net/bluetooth/Kconfig b/net/bluetooth/Kconfig
> index 95d1a66..06c31b9 100644
> --- a/net/bluetooth/Kconfig
> +++ b/net/bluetooth/Kconfig
> @@ -69,6 +69,15 @@ config BT_6LOWPAN
> 	help
> 	  IPv6 compression over Bluetooth Low Energy.
> 
> +config BT_LEDS
> +	bool "Enable LED triggers"
> +	depends on BT
> +	depends on LEDS_CLASS
> +	select LEDS_TRIGGERS
> +	help
> +	  This option selects a few LED triggers for different
> +	  Bluetooth events.
> +
> config BT_SELFTEST
> 	bool "Bluetooth self testing support"
> 	depends on BT && DEBUG_KERNEL
> diff --git a/net/bluetooth/Makefile b/net/bluetooth/Makefile
> index 2b15ae8..b3ff12e 100644
> --- a/net/bluetooth/Makefile
> +++ b/net/bluetooth/Makefile
> @@ -17,6 +17,7 @@ bluetooth-y := af_bluetooth.o hci_core.o hci_conn.o hci_event.o mgmt.o \
> 
> bluetooth-$(CONFIG_BT_BREDR) += sco.o
> bluetooth-$(CONFIG_BT_HS) += a2mp.o amp.o
> +bluetooth-$(CONFIG_BT_LEDS) += leds.o
> bluetooth-$(CONFIG_BT_DEBUGFS) += hci_debugfs.o
> bluetooth-$(CONFIG_BT_SELFTEST) += selftest.o
> 
> diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c
> index 47bcef7..f472b1d 100644
> --- a/net/bluetooth/hci_core.c
> +++ b/net/bluetooth/hci_core.c
> @@ -40,6 +40,7 @@
> #include "hci_request.h"
> #include "hci_debugfs.h"
> #include "smp.h"
> +#include "leds.h"
> 
> static void hci_rx_work(struct work_struct *work);
> static void hci_cmd_work(struct work_struct *work);
> @@ -1395,6 +1396,7 @@ static int hci_dev_do_open(struct hci_dev *hdev)
> 		hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
> 		set_bit(HCI_UP, &hdev->flags);
> 		hci_sock_dev_event(hdev, HCI_DEV_UP);
> +		hci_led_radio(hdev, true);

Lets name this hci_leds_* etc.

And we might want to consider not using the word radio since we never really have used that word anywhere else in the subsystem.

> 		if (!hci_dev_test_flag(hdev, HCI_SETUP) &&
> 		    !hci_dev_test_flag(hdev, HCI_CONFIG) &&
> 		    !hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
> @@ -1532,6 +1534,8 @@ int hci_dev_do_close(struct hci_dev *hdev)
> 		return 0;
> 	}
> 
> +	hci_led_radio(hdev, false);
> +
> 	/* Flush RX and TX works */
> 	flush_work(&hdev->tx_work);
> 	flush_work(&hdev->rx_work);
> @@ -3067,6 +3071,8 @@ int hci_register_dev(struct hci_dev *hdev)
> 	if (error < 0)
> 		goto err_wqueue;
> 
> +	hci_led_init(hdev);
> +
> 	hdev->rfkill = rfkill_alloc(hdev->name, &hdev->dev,
> 				    RFKILL_TYPE_BLUETOOTH, &hci_rfkill_ops,
> 				    hdev);
> @@ -3128,6 +3134,8 @@ void hci_unregister_dev(struct hci_dev *hdev)
> 
> 	id = hdev->id;
> 
> +	hci_led_exit(hdev);
> +
> 	write_lock(&hci_dev_list_lock);
> 	list_del(&hdev->list);
> 	write_unlock(&hci_dev_list_lock);
> diff --git a/net/bluetooth/leds.c b/net/bluetooth/leds.c
> new file mode 100644
> index 0000000..377f643
> --- /dev/null
> +++ b/net/bluetooth/leds.c
> @@ -0,0 +1,99 @@
> +/*
> + * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 as
> + * published by the Free Software Foundation.
> + */
> +
> +#include <net/bluetooth/bluetooth.h>
> +#include <net/bluetooth/hci_core.h>
> +
> +#include "leds.h"
> +
> +struct hci_basic_led_trigger {
> +	struct led_trigger	led_trigger;
> +	atomic_t		cnt;
> +};
> +
> +#define to_hci_basic_led_trigger(arg) container_of(arg, \
> +					struct hci_basic_led_trigger, \
> +					led_trigger)
> +
> +static void hci_basic_led(struct led_trigger *trig, bool inc)
> +{
> +	struct hci_basic_led_trigger *htrig;
> +	int cnt;
> +
> +	if (!trig)
> +		return;
> +
> +	htrig = to_hci_basic_led_trigger(trig);
> +
> +	if (inc)
> +		cnt = atomic_inc_return(&htrig->cnt);
> +	else
> +		cnt = atomic_dec_return(&htrig->cnt);
> +
> +	WARN_ON(cnt < 0);
> +
> +	led_trigger_event(trig, cnt ? LED_FULL : LED_OFF);
> +}

Explain to me the reason for the counter? The Bluetooth subsystem is already guaranteeing the state of HCI_UP to be 100% synchronized. If have one case where it gets set and another where it gets unset. Meaning your counter will do 0 -> 1 and then 1 -> 0.

> +
> +void hci_led_radio(struct hci_dev *hdev, bool enabled)
> +{
> +	hci_basic_led(hdev->radio_led, enabled);
> +}
> +
> +static void hci_basic_led_activate(struct led_classdev *led_cdev)
> +{
> +	struct hci_basic_led_trigger *htrig;
> +	int cnt;
> +
> +	htrig = to_hci_basic_led_trigger(led_cdev->trigger);
> +	cnt = atomic_read(&htrig->cnt);
> +
> +	led_trigger_event(led_cdev->trigger, cnt ? LED_FULL : LED_OFF);
> +}
> +
> +static struct led_trigger *hci_basic_led_allocate(struct hci_dev *hdev,
> +						  const char *name)
> +{
> +	struct hci_basic_led_trigger *htrig;
> +
> +	htrig =	devm_kzalloc(&hdev->dev, sizeof(*htrig), GFP_KERNEL);
> +	if (!htrig)
> +		return NULL;
> +
> +	atomic_set(&htrig->cnt, 0);
> +	htrig->led_trigger.activate = hci_basic_led_activate;
> +	htrig->led_trigger.name = devm_kasprintf(&hdev->dev, GFP_KERNEL,
> +						 "%s-%s", hdev->name,
> +						 name);
> +	if (!htrig->led_trigger.name)
> +		goto err_alloc;
> +
> +	if (led_trigger_register(&htrig->led_trigger))
> +		goto err_register;
> +
> +	return &htrig->led_trigger;
> +
> +err_register:
> +	devm_kfree(&hdev->dev, (void *)htrig->led_trigger.name);
> +err_alloc:
> +	devm_kfree(&hdev->dev, htrig);
> +	return NULL;
> +}
> +
> +void hci_led_init(struct hci_dev *hdev)
> +{
> +	/* initialize radio_led */
> +	hdev->radio_led = hci_basic_led_allocate(hdev, "radio");
> +}
> +
> +void hci_led_exit(struct hci_dev *hdev)
> +{
> +	if (hdev->radio_led)
> +		led_trigger_unregister(hdev->radio_led);
> +}
> +
> diff --git a/net/bluetooth/leds.h b/net/bluetooth/leds.h
> new file mode 100644
> index 0000000..7578cab
> --- /dev/null
> +++ b/net/bluetooth/leds.h
> @@ -0,0 +1,17 @@
> +/*
> + * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 as
> + * published by the Free Software Foundation.
> + */
> +
> +#ifdef CONFIG_BT_LEDS

Use #if IS_ENABLED(CONFIG_BT_LEDS) like we do for the other options.

> +void hci_led_radio(struct hci_dev *hdev, bool enabled);
> +void hci_led_init(struct hci_dev *hdev);
> +void hci_led_exit(struct hci_dev *hdev);
> +#else
> +static inline void hci_led_radio(struct hci_dev *hdev, bool enabled) {}
> +static inline void hci_led_init(struct hci_dev *hdev) {}
> +static inline void hci_led_exit(struct hci_dev *hdev) {}
> +#endif

Regards

Marcel


^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH v2] bluetooth: core: RfC - add basic LED triggers
  2016-01-04 18:19 ` Marcel Holtmann
@ 2016-01-04 21:24   ` Heiner Kallweit
  2016-01-04 21:28     ` Marcel Holtmann
  0 siblings, 1 reply; 7+ messages in thread
From: Heiner Kallweit @ 2016-01-04 21:24 UTC (permalink / raw)
  To: Marcel Holtmann; +Cc: Johan Hedberg, linux-bluetooth, Johannes Berg

Am 04.01.2016 um 19:19 schrieb Marcel Holtmann:
> Hi Heiner,
Hi Marcel,
> 
>> In the bluetooth subsystem I miss the option to use LEDs to indicate
>> the status. This patch is based on the LED triggers in mac80211 and
>> adds two basic LED triggers (powered up, device connected) plus
>> a config option to enable the BT LED triggers.
>>
>> It works fine on my system however I don't know the bluetooth stack
>> in too much detail and therefore appreciate any comment.
>>
>> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
>> ---
>> v2:
>> - completely rewrote the code to be as generic as possible and minimize
>>  the changes to struct hci_dev
>> - make the details of the LED triggers opaque to struct hci_dev
>> - avoid using ifdef CONFIG_BT_LEDS in hci_core.h
>> - start with the radio trigger only
>> - factor out the generic basic led trigger code to facilitate adding
>>  further basic LED triggers
>> - fix indentation in Kconfig
>> - rename led.[ch] to leds.[ch]
>> ---
>> include/net/bluetooth/hci_core.h |  3 ++
>> net/bluetooth/Kconfig            |  9 ++++
>> net/bluetooth/Makefile           |  1 +
>> net/bluetooth/hci_core.c         |  8 ++++
>> net/bluetooth/leds.c             | 99 ++++++++++++++++++++++++++++++++++++++++
>> net/bluetooth/leds.h             | 17 +++++++
>> 6 files changed, 137 insertions(+)
>> create mode 100644 net/bluetooth/leds.c
>> create mode 100644 net/bluetooth/leds.h
>>
>> diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h
>> index c95e032..4ffec8b 100644
>> --- a/include/net/bluetooth/hci_core.h
>> +++ b/include/net/bluetooth/hci_core.h
>> @@ -25,6 +25,7 @@
>> #ifndef __HCI_CORE_H
>> #define __HCI_CORE_H
>>
>> +#include <linux/leds.h>
>> #include <net/bluetooth/hci.h>
>> #include <net/bluetooth/hci_sock.h>
>>
>> @@ -395,6 +396,8 @@ struct hci_dev {
>> 	struct delayed_work	rpa_expired;
>> 	bdaddr_t		rpa;
>>
>> +	struct led_trigger *radio_led;
>> +
> 
> I wonder if this one might not be better named power_led instead.
> 
OK

>> 	int (*open)(struct hci_dev *hdev);
>> 	int (*close)(struct hci_dev *hdev);
>> 	int (*flush)(struct hci_dev *hdev);
>> diff --git a/net/bluetooth/Kconfig b/net/bluetooth/Kconfig
>> index 95d1a66..06c31b9 100644
>> --- a/net/bluetooth/Kconfig
>> +++ b/net/bluetooth/Kconfig
>> @@ -69,6 +69,15 @@ config BT_6LOWPAN
>> 	help
>> 	  IPv6 compression over Bluetooth Low Energy.
>>
>> +config BT_LEDS
>> +	bool "Enable LED triggers"
>> +	depends on BT
>> +	depends on LEDS_CLASS
>> +	select LEDS_TRIGGERS
>> +	help
>> +	  This option selects a few LED triggers for different
>> +	  Bluetooth events.
>> +
>> config BT_SELFTEST
>> 	bool "Bluetooth self testing support"
>> 	depends on BT && DEBUG_KERNEL
>> diff --git a/net/bluetooth/Makefile b/net/bluetooth/Makefile
>> index 2b15ae8..b3ff12e 100644
>> --- a/net/bluetooth/Makefile
>> +++ b/net/bluetooth/Makefile
>> @@ -17,6 +17,7 @@ bluetooth-y := af_bluetooth.o hci_core.o hci_conn.o hci_event.o mgmt.o \
>>
>> bluetooth-$(CONFIG_BT_BREDR) += sco.o
>> bluetooth-$(CONFIG_BT_HS) += a2mp.o amp.o
>> +bluetooth-$(CONFIG_BT_LEDS) += leds.o
>> bluetooth-$(CONFIG_BT_DEBUGFS) += hci_debugfs.o
>> bluetooth-$(CONFIG_BT_SELFTEST) += selftest.o
>>
>> diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c
>> index 47bcef7..f472b1d 100644
>> --- a/net/bluetooth/hci_core.c
>> +++ b/net/bluetooth/hci_core.c
>> @@ -40,6 +40,7 @@
>> #include "hci_request.h"
>> #include "hci_debugfs.h"
>> #include "smp.h"
>> +#include "leds.h"
>>
>> static void hci_rx_work(struct work_struct *work);
>> static void hci_cmd_work(struct work_struct *work);
>> @@ -1395,6 +1396,7 @@ static int hci_dev_do_open(struct hci_dev *hdev)
>> 		hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
>> 		set_bit(HCI_UP, &hdev->flags);
>> 		hci_sock_dev_event(hdev, HCI_DEV_UP);
>> +		hci_led_radio(hdev, true);
> 
> Lets name this hci_leds_* etc.
> 
> And we might want to consider not using the word radio since we never really have used that word anywhere else in the subsystem.
> 
Then let's use the word power here too (e.g. hci_leds_powered).

>> 		if (!hci_dev_test_flag(hdev, HCI_SETUP) &&
>> 		    !hci_dev_test_flag(hdev, HCI_CONFIG) &&
>> 		    !hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
>> @@ -1532,6 +1534,8 @@ int hci_dev_do_close(struct hci_dev *hdev)
>> 		return 0;
>> 	}
>>
>> +	hci_led_radio(hdev, false);
>> +
>> 	/* Flush RX and TX works */
>> 	flush_work(&hdev->tx_work);
>> 	flush_work(&hdev->rx_work);
>> @@ -3067,6 +3071,8 @@ int hci_register_dev(struct hci_dev *hdev)
>> 	if (error < 0)
>> 		goto err_wqueue;
>>
>> +	hci_led_init(hdev);
>> +
>> 	hdev->rfkill = rfkill_alloc(hdev->name, &hdev->dev,
>> 				    RFKILL_TYPE_BLUETOOTH, &hci_rfkill_ops,
>> 				    hdev);
>> @@ -3128,6 +3134,8 @@ void hci_unregister_dev(struct hci_dev *hdev)
>>
>> 	id = hdev->id;
>>
>> +	hci_led_exit(hdev);
>> +
>> 	write_lock(&hci_dev_list_lock);
>> 	list_del(&hdev->list);
>> 	write_unlock(&hci_dev_list_lock);
>> diff --git a/net/bluetooth/leds.c b/net/bluetooth/leds.c
>> new file mode 100644
>> index 0000000..377f643
>> --- /dev/null
>> +++ b/net/bluetooth/leds.c
>> @@ -0,0 +1,99 @@
>> +/*
>> + * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
>> + *
>> + * This program is free software; you can redistribute it and/or modify
>> + * it under the terms of the GNU General Public License version 2 as
>> + * published by the Free Software Foundation.
>> + */
>> +
>> +#include <net/bluetooth/bluetooth.h>
>> +#include <net/bluetooth/hci_core.h>
>> +
>> +#include "leds.h"
>> +
>> +struct hci_basic_led_trigger {
>> +	struct led_trigger	led_trigger;
>> +	atomic_t		cnt;
>> +};
>> +
>> +#define to_hci_basic_led_trigger(arg) container_of(arg, \
>> +					struct hci_basic_led_trigger, \
>> +					led_trigger)
>> +
>> +static void hci_basic_led(struct led_trigger *trig, bool inc)
>> +{
>> +	struct hci_basic_led_trigger *htrig;
>> +	int cnt;
>> +
>> +	if (!trig)
>> +		return;
>> +
>> +	htrig = to_hci_basic_led_trigger(trig);
>> +
>> +	if (inc)
>> +		cnt = atomic_inc_return(&htrig->cnt);
>> +	else
>> +		cnt = atomic_dec_return(&htrig->cnt);
>> +
>> +	WARN_ON(cnt < 0);
>> +
>> +	led_trigger_event(trig, cnt ? LED_FULL : LED_OFF);
>> +}
> 
> Explain to me the reason for the counter? The Bluetooth subsystem is already guaranteeing the state of HCI_UP to be 100% synchronized. If have one case where it gets set and another where it gets unset. Meaning your counter will do 0 -> 1 and then 1 -> 0.
> 
Indeed the counter is not needed in this particular case.
My intention was to make this basic led class more generic so that it can be re-used for triggers
(to be added later) based on use cases like "at least one connection" where counting may be needed.

>> +
>> +void hci_led_radio(struct hci_dev *hdev, bool enabled)
>> +{
>> +	hci_basic_led(hdev->radio_led, enabled);
>> +}
>> +
>> +static void hci_basic_led_activate(struct led_classdev *led_cdev)
>> +{
>> +	struct hci_basic_led_trigger *htrig;
>> +	int cnt;
>> +
>> +	htrig = to_hci_basic_led_trigger(led_cdev->trigger);
>> +	cnt = atomic_read(&htrig->cnt);
>> +
>> +	led_trigger_event(led_cdev->trigger, cnt ? LED_FULL : LED_OFF);
>> +}
>> +
>> +static struct led_trigger *hci_basic_led_allocate(struct hci_dev *hdev,
>> +						  const char *name)
>> +{
>> +	struct hci_basic_led_trigger *htrig;
>> +
>> +	htrig =	devm_kzalloc(&hdev->dev, sizeof(*htrig), GFP_KERNEL);
>> +	if (!htrig)
>> +		return NULL;
>> +
>> +	atomic_set(&htrig->cnt, 0);
>> +	htrig->led_trigger.activate = hci_basic_led_activate;
>> +	htrig->led_trigger.name = devm_kasprintf(&hdev->dev, GFP_KERNEL,
>> +						 "%s-%s", hdev->name,
>> +						 name);
>> +	if (!htrig->led_trigger.name)
>> +		goto err_alloc;
>> +
>> +	if (led_trigger_register(&htrig->led_trigger))
>> +		goto err_register;
>> +
>> +	return &htrig->led_trigger;
>> +
>> +err_register:
>> +	devm_kfree(&hdev->dev, (void *)htrig->led_trigger.name);
>> +err_alloc:
>> +	devm_kfree(&hdev->dev, htrig);
>> +	return NULL;
>> +}
>> +
>> +void hci_led_init(struct hci_dev *hdev)
>> +{
>> +	/* initialize radio_led */
>> +	hdev->radio_led = hci_basic_led_allocate(hdev, "radio");
>> +}
>> +
>> +void hci_led_exit(struct hci_dev *hdev)
>> +{
>> +	if (hdev->radio_led)
>> +		led_trigger_unregister(hdev->radio_led);
>> +}
>> +
>> diff --git a/net/bluetooth/leds.h b/net/bluetooth/leds.h
>> new file mode 100644
>> index 0000000..7578cab
>> --- /dev/null
>> +++ b/net/bluetooth/leds.h
>> @@ -0,0 +1,17 @@
>> +/*
>> + * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
>> + *
>> + * This program is free software; you can redistribute it and/or modify
>> + * it under the terms of the GNU General Public License version 2 as
>> + * published by the Free Software Foundation.
>> + */
>> +
>> +#ifdef CONFIG_BT_LEDS
> 
> Use #if IS_ENABLED(CONFIG_BT_LEDS) like we do for the other options.
> 
OK

>> +void hci_led_radio(struct hci_dev *hdev, bool enabled);
>> +void hci_led_init(struct hci_dev *hdev);
>> +void hci_led_exit(struct hci_dev *hdev);
>> +#else
>> +static inline void hci_led_radio(struct hci_dev *hdev, bool enabled) {}
>> +static inline void hci_led_init(struct hci_dev *hdev) {}
>> +static inline void hci_led_exit(struct hci_dev *hdev) {}
>> +#endif
> 
> Regards
> 
> Marcel
> 
Thanks, Heiner
> 

^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH v2] bluetooth: core: RfC - add basic LED triggers
  2016-01-04 21:24   ` Heiner Kallweit
@ 2016-01-04 21:28     ` Marcel Holtmann
  2016-01-05  6:50       ` Heiner Kallweit
  0 siblings, 1 reply; 7+ messages in thread
From: Marcel Holtmann @ 2016-01-04 21:28 UTC (permalink / raw)
  To: Heiner Kallweit; +Cc: Johan Hedberg, BlueZ development, Johannes Berg

Hi Heiner,

>>> In the bluetooth subsystem I miss the option to use LEDs to indicate
>>> the status. This patch is based on the LED triggers in mac80211 and
>>> adds two basic LED triggers (powered up, device connected) plus
>>> a config option to enable the BT LED triggers.
>>> 
>>> It works fine on my system however I don't know the bluetooth stack
>>> in too much detail and therefore appreciate any comment.
>>> 
>>> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
>>> ---
>>> v2:
>>> - completely rewrote the code to be as generic as possible and minimize
>>> the changes to struct hci_dev
>>> - make the details of the LED triggers opaque to struct hci_dev
>>> - avoid using ifdef CONFIG_BT_LEDS in hci_core.h
>>> - start with the radio trigger only
>>> - factor out the generic basic led trigger code to facilitate adding
>>> further basic LED triggers
>>> - fix indentation in Kconfig
>>> - rename led.[ch] to leds.[ch]
>>> ---
>>> include/net/bluetooth/hci_core.h |  3 ++
>>> net/bluetooth/Kconfig            |  9 ++++
>>> net/bluetooth/Makefile           |  1 +
>>> net/bluetooth/hci_core.c         |  8 ++++
>>> net/bluetooth/leds.c             | 99 ++++++++++++++++++++++++++++++++++++++++
>>> net/bluetooth/leds.h             | 17 +++++++
>>> 6 files changed, 137 insertions(+)
>>> create mode 100644 net/bluetooth/leds.c
>>> create mode 100644 net/bluetooth/leds.h
>>> 
>>> diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h
>>> index c95e032..4ffec8b 100644
>>> --- a/include/net/bluetooth/hci_core.h
>>> +++ b/include/net/bluetooth/hci_core.h
>>> @@ -25,6 +25,7 @@
>>> #ifndef __HCI_CORE_H
>>> #define __HCI_CORE_H
>>> 
>>> +#include <linux/leds.h>
>>> #include <net/bluetooth/hci.h>
>>> #include <net/bluetooth/hci_sock.h>
>>> 
>>> @@ -395,6 +396,8 @@ struct hci_dev {
>>> 	struct delayed_work	rpa_expired;
>>> 	bdaddr_t		rpa;
>>> 
>>> +	struct led_trigger *radio_led;
>>> +
>> 
>> I wonder if this one might not be better named power_led instead.
>> 
> OK
> 
>>> 	int (*open)(struct hci_dev *hdev);
>>> 	int (*close)(struct hci_dev *hdev);
>>> 	int (*flush)(struct hci_dev *hdev);
>>> diff --git a/net/bluetooth/Kconfig b/net/bluetooth/Kconfig
>>> index 95d1a66..06c31b9 100644
>>> --- a/net/bluetooth/Kconfig
>>> +++ b/net/bluetooth/Kconfig
>>> @@ -69,6 +69,15 @@ config BT_6LOWPAN
>>> 	help
>>> 	  IPv6 compression over Bluetooth Low Energy.
>>> 
>>> +config BT_LEDS
>>> +	bool "Enable LED triggers"
>>> +	depends on BT
>>> +	depends on LEDS_CLASS
>>> +	select LEDS_TRIGGERS
>>> +	help
>>> +	  This option selects a few LED triggers for different
>>> +	  Bluetooth events.
>>> +
>>> config BT_SELFTEST
>>> 	bool "Bluetooth self testing support"
>>> 	depends on BT && DEBUG_KERNEL
>>> diff --git a/net/bluetooth/Makefile b/net/bluetooth/Makefile
>>> index 2b15ae8..b3ff12e 100644
>>> --- a/net/bluetooth/Makefile
>>> +++ b/net/bluetooth/Makefile
>>> @@ -17,6 +17,7 @@ bluetooth-y := af_bluetooth.o hci_core.o hci_conn.o hci_event.o mgmt.o \
>>> 
>>> bluetooth-$(CONFIG_BT_BREDR) += sco.o
>>> bluetooth-$(CONFIG_BT_HS) += a2mp.o amp.o
>>> +bluetooth-$(CONFIG_BT_LEDS) += leds.o
>>> bluetooth-$(CONFIG_BT_DEBUGFS) += hci_debugfs.o
>>> bluetooth-$(CONFIG_BT_SELFTEST) += selftest.o
>>> 
>>> diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c
>>> index 47bcef7..f472b1d 100644
>>> --- a/net/bluetooth/hci_core.c
>>> +++ b/net/bluetooth/hci_core.c
>>> @@ -40,6 +40,7 @@
>>> #include "hci_request.h"
>>> #include "hci_debugfs.h"
>>> #include "smp.h"
>>> +#include "leds.h"
>>> 
>>> static void hci_rx_work(struct work_struct *work);
>>> static void hci_cmd_work(struct work_struct *work);
>>> @@ -1395,6 +1396,7 @@ static int hci_dev_do_open(struct hci_dev *hdev)
>>> 		hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
>>> 		set_bit(HCI_UP, &hdev->flags);
>>> 		hci_sock_dev_event(hdev, HCI_DEV_UP);
>>> +		hci_led_radio(hdev, true);
>> 
>> Lets name this hci_leds_* etc.
>> 
>> And we might want to consider not using the word radio since we never really have used that word anywhere else in the subsystem.
>> 
> Then let's use the word power here too (e.g. hci_leds_powered).

I am not 100% sure that is the best name either, but it has at least better meaning than radio.

> 
>>> 		if (!hci_dev_test_flag(hdev, HCI_SETUP) &&
>>> 		    !hci_dev_test_flag(hdev, HCI_CONFIG) &&
>>> 		    !hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
>>> @@ -1532,6 +1534,8 @@ int hci_dev_do_close(struct hci_dev *hdev)
>>> 		return 0;
>>> 	}
>>> 
>>> +	hci_led_radio(hdev, false);
>>> +
>>> 	/* Flush RX and TX works */
>>> 	flush_work(&hdev->tx_work);
>>> 	flush_work(&hdev->rx_work);
>>> @@ -3067,6 +3071,8 @@ int hci_register_dev(struct hci_dev *hdev)
>>> 	if (error < 0)
>>> 		goto err_wqueue;
>>> 
>>> +	hci_led_init(hdev);
>>> +
>>> 	hdev->rfkill = rfkill_alloc(hdev->name, &hdev->dev,
>>> 				    RFKILL_TYPE_BLUETOOTH, &hci_rfkill_ops,
>>> 				    hdev);
>>> @@ -3128,6 +3134,8 @@ void hci_unregister_dev(struct hci_dev *hdev)
>>> 
>>> 	id = hdev->id;
>>> 
>>> +	hci_led_exit(hdev);
>>> +
>>> 	write_lock(&hci_dev_list_lock);
>>> 	list_del(&hdev->list);
>>> 	write_unlock(&hci_dev_list_lock);
>>> diff --git a/net/bluetooth/leds.c b/net/bluetooth/leds.c
>>> new file mode 100644
>>> index 0000000..377f643
>>> --- /dev/null
>>> +++ b/net/bluetooth/leds.c
>>> @@ -0,0 +1,99 @@
>>> +/*
>>> + * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
>>> + *
>>> + * This program is free software; you can redistribute it and/or modify
>>> + * it under the terms of the GNU General Public License version 2 as
>>> + * published by the Free Software Foundation.
>>> + */
>>> +
>>> +#include <net/bluetooth/bluetooth.h>
>>> +#include <net/bluetooth/hci_core.h>
>>> +
>>> +#include "leds.h"
>>> +
>>> +struct hci_basic_led_trigger {
>>> +	struct led_trigger	led_trigger;
>>> +	atomic_t		cnt;
>>> +};
>>> +
>>> +#define to_hci_basic_led_trigger(arg) container_of(arg, \
>>> +					struct hci_basic_led_trigger, \
>>> +					led_trigger)
>>> +
>>> +static void hci_basic_led(struct led_trigger *trig, bool inc)
>>> +{
>>> +	struct hci_basic_led_trigger *htrig;
>>> +	int cnt;
>>> +
>>> +	if (!trig)
>>> +		return;
>>> +
>>> +	htrig = to_hci_basic_led_trigger(trig);
>>> +
>>> +	if (inc)
>>> +		cnt = atomic_inc_return(&htrig->cnt);
>>> +	else
>>> +		cnt = atomic_dec_return(&htrig->cnt);
>>> +
>>> +	WARN_ON(cnt < 0);
>>> +
>>> +	led_trigger_event(trig, cnt ? LED_FULL : LED_OFF);
>>> +}
>> 
>> Explain to me the reason for the counter? The Bluetooth subsystem is already guaranteeing the state of HCI_UP to be 100% synchronized. If have one case where it gets set and another where it gets unset. Meaning your counter will do 0 -> 1 and then 1 -> 0.
>> 
> Indeed the counter is not needed in this particular case.
> My intention was to make this basic led class more generic so that it can be re-used for triggers
> (to be added later) based on use cases like "at least one connection" where counting may be needed.

Actually it isn't needed there either. We have a hci_conn_hash that does the counting for you already. You pretty much just have to sum it up correctly and you know the correct state of your led.

Regards

Marcel


^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH v2] bluetooth: core: RfC - add basic LED triggers
  2016-01-04 21:28     ` Marcel Holtmann
@ 2016-01-05  6:50       ` Heiner Kallweit
  2016-01-05  7:03         ` Marcel Holtmann
  0 siblings, 1 reply; 7+ messages in thread
From: Heiner Kallweit @ 2016-01-05  6:50 UTC (permalink / raw)
  To: Marcel Holtmann; +Cc: Johan Hedberg, BlueZ development, Johannes Berg

Am 04.01.2016 um 22:28 schrieb Marcel Holtmann:
> Hi Heiner,
> 
>>>> In the bluetooth subsystem I miss the option to use LEDs to indicate
>>>> the status. This patch is based on the LED triggers in mac80211 and
>>>> adds two basic LED triggers (powered up, device connected) plus
>>>> a config option to enable the BT LED triggers.
>>>>
>>>> It works fine on my system however I don't know the bluetooth stack
>>>> in too much detail and therefore appreciate any comment.
>>>>
>>>> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
>>>> ---
>>>> v2:
>>>> - completely rewrote the code to be as generic as possible and minimize
>>>> the changes to struct hci_dev
>>>> - make the details of the LED triggers opaque to struct hci_dev
>>>> - avoid using ifdef CONFIG_BT_LEDS in hci_core.h
>>>> - start with the radio trigger only
>>>> - factor out the generic basic led trigger code to facilitate adding
>>>> further basic LED triggers
>>>> - fix indentation in Kconfig
>>>> - rename led.[ch] to leds.[ch]
>>>> ---
>>>> include/net/bluetooth/hci_core.h |  3 ++
>>>> net/bluetooth/Kconfig            |  9 ++++
>>>> net/bluetooth/Makefile           |  1 +
>>>> net/bluetooth/hci_core.c         |  8 ++++
>>>> net/bluetooth/leds.c             | 99 ++++++++++++++++++++++++++++++++++++++++
>>>> net/bluetooth/leds.h             | 17 +++++++
>>>> 6 files changed, 137 insertions(+)
>>>> create mode 100644 net/bluetooth/leds.c
>>>> create mode 100644 net/bluetooth/leds.h
>>>>
>>>> diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h
>>>> index c95e032..4ffec8b 100644
>>>> --- a/include/net/bluetooth/hci_core.h
>>>> +++ b/include/net/bluetooth/hci_core.h
>>>> @@ -25,6 +25,7 @@
>>>> #ifndef __HCI_CORE_H
>>>> #define __HCI_CORE_H
>>>>
>>>> +#include <linux/leds.h>
>>>> #include <net/bluetooth/hci.h>
>>>> #include <net/bluetooth/hci_sock.h>
>>>>
>>>> @@ -395,6 +396,8 @@ struct hci_dev {
>>>> 	struct delayed_work	rpa_expired;
>>>> 	bdaddr_t		rpa;
>>>>
>>>> +	struct led_trigger *radio_led;
>>>> +
>>>
>>> I wonder if this one might not be better named power_led instead.
>>>
>> OK
>>
>>>> 	int (*open)(struct hci_dev *hdev);
>>>> 	int (*close)(struct hci_dev *hdev);
>>>> 	int (*flush)(struct hci_dev *hdev);
>>>> diff --git a/net/bluetooth/Kconfig b/net/bluetooth/Kconfig
>>>> index 95d1a66..06c31b9 100644
>>>> --- a/net/bluetooth/Kconfig
>>>> +++ b/net/bluetooth/Kconfig
>>>> @@ -69,6 +69,15 @@ config BT_6LOWPAN
>>>> 	help
>>>> 	  IPv6 compression over Bluetooth Low Energy.
>>>>
>>>> +config BT_LEDS
>>>> +	bool "Enable LED triggers"
>>>> +	depends on BT
>>>> +	depends on LEDS_CLASS
>>>> +	select LEDS_TRIGGERS
>>>> +	help
>>>> +	  This option selects a few LED triggers for different
>>>> +	  Bluetooth events.
>>>> +
>>>> config BT_SELFTEST
>>>> 	bool "Bluetooth self testing support"
>>>> 	depends on BT && DEBUG_KERNEL
>>>> diff --git a/net/bluetooth/Makefile b/net/bluetooth/Makefile
>>>> index 2b15ae8..b3ff12e 100644
>>>> --- a/net/bluetooth/Makefile
>>>> +++ b/net/bluetooth/Makefile
>>>> @@ -17,6 +17,7 @@ bluetooth-y := af_bluetooth.o hci_core.o hci_conn.o hci_event.o mgmt.o \
>>>>
>>>> bluetooth-$(CONFIG_BT_BREDR) += sco.o
>>>> bluetooth-$(CONFIG_BT_HS) += a2mp.o amp.o
>>>> +bluetooth-$(CONFIG_BT_LEDS) += leds.o
>>>> bluetooth-$(CONFIG_BT_DEBUGFS) += hci_debugfs.o
>>>> bluetooth-$(CONFIG_BT_SELFTEST) += selftest.o
>>>>
>>>> diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c
>>>> index 47bcef7..f472b1d 100644
>>>> --- a/net/bluetooth/hci_core.c
>>>> +++ b/net/bluetooth/hci_core.c
>>>> @@ -40,6 +40,7 @@
>>>> #include "hci_request.h"
>>>> #include "hci_debugfs.h"
>>>> #include "smp.h"
>>>> +#include "leds.h"
>>>>
>>>> static void hci_rx_work(struct work_struct *work);
>>>> static void hci_cmd_work(struct work_struct *work);
>>>> @@ -1395,6 +1396,7 @@ static int hci_dev_do_open(struct hci_dev *hdev)
>>>> 		hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
>>>> 		set_bit(HCI_UP, &hdev->flags);
>>>> 		hci_sock_dev_event(hdev, HCI_DEV_UP);
>>>> +		hci_led_radio(hdev, true);
>>>
>>> Lets name this hci_leds_* etc.
>>>
>>> And we might want to consider not using the word radio since we never really have used that word anywhere else in the subsystem.
>>>
>> Then let's use the word power here too (e.g. hci_leds_powered).
> 
> I am not 100% sure that is the best name either, but it has at least better meaning than radio.
> 
Hmm, maybe hci_leds_ispowered ?
>>
>>>> 		if (!hci_dev_test_flag(hdev, HCI_SETUP) &&
>>>> 		    !hci_dev_test_flag(hdev, HCI_CONFIG) &&
>>>> 		    !hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
>>>> @@ -1532,6 +1534,8 @@ int hci_dev_do_close(struct hci_dev *hdev)
>>>> 		return 0;
>>>> 	}
>>>>
>>>> +	hci_led_radio(hdev, false);
>>>> +
>>>> 	/* Flush RX and TX works */
>>>> 	flush_work(&hdev->tx_work);
>>>> 	flush_work(&hdev->rx_work);
>>>> @@ -3067,6 +3071,8 @@ int hci_register_dev(struct hci_dev *hdev)
>>>> 	if (error < 0)
>>>> 		goto err_wqueue;
>>>>
>>>> +	hci_led_init(hdev);
>>>> +
>>>> 	hdev->rfkill = rfkill_alloc(hdev->name, &hdev->dev,
>>>> 				    RFKILL_TYPE_BLUETOOTH, &hci_rfkill_ops,
>>>> 				    hdev);
>>>> @@ -3128,6 +3134,8 @@ void hci_unregister_dev(struct hci_dev *hdev)
>>>>
>>>> 	id = hdev->id;
>>>>
>>>> +	hci_led_exit(hdev);
>>>> +
>>>> 	write_lock(&hci_dev_list_lock);
>>>> 	list_del(&hdev->list);
>>>> 	write_unlock(&hci_dev_list_lock);
>>>> diff --git a/net/bluetooth/leds.c b/net/bluetooth/leds.c
>>>> new file mode 100644
>>>> index 0000000..377f643
>>>> --- /dev/null
>>>> +++ b/net/bluetooth/leds.c
>>>> @@ -0,0 +1,99 @@
>>>> +/*
>>>> + * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
>>>> + *
>>>> + * This program is free software; you can redistribute it and/or modify
>>>> + * it under the terms of the GNU General Public License version 2 as
>>>> + * published by the Free Software Foundation.
>>>> + */
>>>> +
>>>> +#include <net/bluetooth/bluetooth.h>
>>>> +#include <net/bluetooth/hci_core.h>
>>>> +
>>>> +#include "leds.h"
>>>> +
>>>> +struct hci_basic_led_trigger {
>>>> +	struct led_trigger	led_trigger;
>>>> +	atomic_t		cnt;
>>>> +};
>>>> +
>>>> +#define to_hci_basic_led_trigger(arg) container_of(arg, \
>>>> +					struct hci_basic_led_trigger, \
>>>> +					led_trigger)
>>>> +
>>>> +static void hci_basic_led(struct led_trigger *trig, bool inc)
>>>> +{
>>>> +	struct hci_basic_led_trigger *htrig;
>>>> +	int cnt;
>>>> +
>>>> +	if (!trig)
>>>> +		return;
>>>> +
>>>> +	htrig = to_hci_basic_led_trigger(trig);
>>>> +
>>>> +	if (inc)
>>>> +		cnt = atomic_inc_return(&htrig->cnt);
>>>> +	else
>>>> +		cnt = atomic_dec_return(&htrig->cnt);
>>>> +
>>>> +	WARN_ON(cnt < 0);
>>>> +
>>>> +	led_trigger_event(trig, cnt ? LED_FULL : LED_OFF);
>>>> +}
>>>
>>> Explain to me the reason for the counter? The Bluetooth subsystem is already guaranteeing the state of HCI_UP to be 100% synchronized. If have one case where it gets set and another where it gets unset. Meaning your counter will do 0 -> 1 and then 1 -> 0.
>>>
>> Indeed the counter is not needed in this particular case.
>> My intention was to make this basic led class more generic so that it can be re-used for triggers
>> (to be added later) based on use cases like "at least one connection" where counting may be needed.
> 
> Actually it isn't needed there either. We have a hci_conn_hash that does the counting for you already. You pretty much just have to sum it up correctly and you know the correct state of your led.
> 
OK. We just might have one remaining issue: When the trigger gets activated, how to find out whether hci currently is powered or not?
We can use the counter (as is) or we could store hdev in our trigger class and use hdev_is_powered(hdev) from within the activate callback.
Would you prefer the latter approach?

> Regards
> 
> Marcel
> 
Heiner

^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH v2] bluetooth: core: RfC - add basic LED triggers
  2016-01-05  6:50       ` Heiner Kallweit
@ 2016-01-05  7:03         ` Marcel Holtmann
  2016-01-05 20:12           ` Heiner Kallweit
  0 siblings, 1 reply; 7+ messages in thread
From: Marcel Holtmann @ 2016-01-05  7:03 UTC (permalink / raw)
  To: Heiner Kallweit; +Cc: Johan Hedberg, BlueZ development, Johannes Berg

Hi Heiner,

>>>>> In the bluetooth subsystem I miss the option to use LEDs to indicate
>>>>> the status. This patch is based on the LED triggers in mac80211 and
>>>>> adds two basic LED triggers (powered up, device connected) plus
>>>>> a config option to enable the BT LED triggers.
>>>>> 
>>>>> It works fine on my system however I don't know the bluetooth stack
>>>>> in too much detail and therefore appreciate any comment.
>>>>> 
>>>>> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
>>>>> ---
>>>>> v2:
>>>>> - completely rewrote the code to be as generic as possible and minimize
>>>>> the changes to struct hci_dev
>>>>> - make the details of the LED triggers opaque to struct hci_dev
>>>>> - avoid using ifdef CONFIG_BT_LEDS in hci_core.h
>>>>> - start with the radio trigger only
>>>>> - factor out the generic basic led trigger code to facilitate adding
>>>>> further basic LED triggers
>>>>> - fix indentation in Kconfig
>>>>> - rename led.[ch] to leds.[ch]
>>>>> ---
>>>>> include/net/bluetooth/hci_core.h |  3 ++
>>>>> net/bluetooth/Kconfig            |  9 ++++
>>>>> net/bluetooth/Makefile           |  1 +
>>>>> net/bluetooth/hci_core.c         |  8 ++++
>>>>> net/bluetooth/leds.c             | 99 ++++++++++++++++++++++++++++++++++++++++
>>>>> net/bluetooth/leds.h             | 17 +++++++
>>>>> 6 files changed, 137 insertions(+)
>>>>> create mode 100644 net/bluetooth/leds.c
>>>>> create mode 100644 net/bluetooth/leds.h
>>>>> 
>>>>> diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h
>>>>> index c95e032..4ffec8b 100644
>>>>> --- a/include/net/bluetooth/hci_core.h
>>>>> +++ b/include/net/bluetooth/hci_core.h
>>>>> @@ -25,6 +25,7 @@
>>>>> #ifndef __HCI_CORE_H
>>>>> #define __HCI_CORE_H
>>>>> 
>>>>> +#include <linux/leds.h>
>>>>> #include <net/bluetooth/hci.h>
>>>>> #include <net/bluetooth/hci_sock.h>
>>>>> 
>>>>> @@ -395,6 +396,8 @@ struct hci_dev {
>>>>> 	struct delayed_work	rpa_expired;
>>>>> 	bdaddr_t		rpa;
>>>>> 
>>>>> +	struct led_trigger *radio_led;
>>>>> +
>>>> 
>>>> I wonder if this one might not be better named power_led instead.
>>>> 
>>> OK
>>> 
>>>>> 	int (*open)(struct hci_dev *hdev);
>>>>> 	int (*close)(struct hci_dev *hdev);
>>>>> 	int (*flush)(struct hci_dev *hdev);
>>>>> diff --git a/net/bluetooth/Kconfig b/net/bluetooth/Kconfig
>>>>> index 95d1a66..06c31b9 100644
>>>>> --- a/net/bluetooth/Kconfig
>>>>> +++ b/net/bluetooth/Kconfig
>>>>> @@ -69,6 +69,15 @@ config BT_6LOWPAN
>>>>> 	help
>>>>> 	  IPv6 compression over Bluetooth Low Energy.
>>>>> 
>>>>> +config BT_LEDS
>>>>> +	bool "Enable LED triggers"
>>>>> +	depends on BT
>>>>> +	depends on LEDS_CLASS
>>>>> +	select LEDS_TRIGGERS
>>>>> +	help
>>>>> +	  This option selects a few LED triggers for different
>>>>> +	  Bluetooth events.
>>>>> +
>>>>> config BT_SELFTEST
>>>>> 	bool "Bluetooth self testing support"
>>>>> 	depends on BT && DEBUG_KERNEL
>>>>> diff --git a/net/bluetooth/Makefile b/net/bluetooth/Makefile
>>>>> index 2b15ae8..b3ff12e 100644
>>>>> --- a/net/bluetooth/Makefile
>>>>> +++ b/net/bluetooth/Makefile
>>>>> @@ -17,6 +17,7 @@ bluetooth-y := af_bluetooth.o hci_core.o hci_conn.o hci_event.o mgmt.o \
>>>>> 
>>>>> bluetooth-$(CONFIG_BT_BREDR) += sco.o
>>>>> bluetooth-$(CONFIG_BT_HS) += a2mp.o amp.o
>>>>> +bluetooth-$(CONFIG_BT_LEDS) += leds.o
>>>>> bluetooth-$(CONFIG_BT_DEBUGFS) += hci_debugfs.o
>>>>> bluetooth-$(CONFIG_BT_SELFTEST) += selftest.o
>>>>> 
>>>>> diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c
>>>>> index 47bcef7..f472b1d 100644
>>>>> --- a/net/bluetooth/hci_core.c
>>>>> +++ b/net/bluetooth/hci_core.c
>>>>> @@ -40,6 +40,7 @@
>>>>> #include "hci_request.h"
>>>>> #include "hci_debugfs.h"
>>>>> #include "smp.h"
>>>>> +#include "leds.h"
>>>>> 
>>>>> static void hci_rx_work(struct work_struct *work);
>>>>> static void hci_cmd_work(struct work_struct *work);
>>>>> @@ -1395,6 +1396,7 @@ static int hci_dev_do_open(struct hci_dev *hdev)
>>>>> 		hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
>>>>> 		set_bit(HCI_UP, &hdev->flags);
>>>>> 		hci_sock_dev_event(hdev, HCI_DEV_UP);
>>>>> +		hci_led_radio(hdev, true);
>>>> 
>>>> Lets name this hci_leds_* etc.
>>>> 
>>>> And we might want to consider not using the word radio since we never really have used that word anywhere else in the subsystem.
>>>> 
>>> Then let's use the word power here too (e.g. hci_leds_powered).
>> 
>> I am not 100% sure that is the best name either, but it has at least better meaning than radio.
>> 
> Hmm, maybe hci_leds_ispowered ?

That is more like a query that should return true/false. Maybe hci_leds_update_powered.

>>> 
>>>>> 		if (!hci_dev_test_flag(hdev, HCI_SETUP) &&
>>>>> 		    !hci_dev_test_flag(hdev, HCI_CONFIG) &&
>>>>> 		    !hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
>>>>> @@ -1532,6 +1534,8 @@ int hci_dev_do_close(struct hci_dev *hdev)
>>>>> 		return 0;
>>>>> 	}
>>>>> 
>>>>> +	hci_led_radio(hdev, false);
>>>>> +
>>>>> 	/* Flush RX and TX works */
>>>>> 	flush_work(&hdev->tx_work);
>>>>> 	flush_work(&hdev->rx_work);
>>>>> @@ -3067,6 +3071,8 @@ int hci_register_dev(struct hci_dev *hdev)
>>>>> 	if (error < 0)
>>>>> 		goto err_wqueue;
>>>>> 
>>>>> +	hci_led_init(hdev);
>>>>> +
>>>>> 	hdev->rfkill = rfkill_alloc(hdev->name, &hdev->dev,
>>>>> 				    RFKILL_TYPE_BLUETOOTH, &hci_rfkill_ops,
>>>>> 				    hdev);
>>>>> @@ -3128,6 +3134,8 @@ void hci_unregister_dev(struct hci_dev *hdev)
>>>>> 
>>>>> 	id = hdev->id;
>>>>> 
>>>>> +	hci_led_exit(hdev);
>>>>> +
>>>>> 	write_lock(&hci_dev_list_lock);
>>>>> 	list_del(&hdev->list);
>>>>> 	write_unlock(&hci_dev_list_lock);
>>>>> diff --git a/net/bluetooth/leds.c b/net/bluetooth/leds.c
>>>>> new file mode 100644
>>>>> index 0000000..377f643
>>>>> --- /dev/null
>>>>> +++ b/net/bluetooth/leds.c
>>>>> @@ -0,0 +1,99 @@
>>>>> +/*
>>>>> + * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
>>>>> + *
>>>>> + * This program is free software; you can redistribute it and/or modify
>>>>> + * it under the terms of the GNU General Public License version 2 as
>>>>> + * published by the Free Software Foundation.
>>>>> + */
>>>>> +
>>>>> +#include <net/bluetooth/bluetooth.h>
>>>>> +#include <net/bluetooth/hci_core.h>
>>>>> +
>>>>> +#include "leds.h"
>>>>> +
>>>>> +struct hci_basic_led_trigger {
>>>>> +	struct led_trigger	led_trigger;
>>>>> +	atomic_t		cnt;
>>>>> +};
>>>>> +
>>>>> +#define to_hci_basic_led_trigger(arg) container_of(arg, \
>>>>> +					struct hci_basic_led_trigger, \
>>>>> +					led_trigger)
>>>>> +
>>>>> +static void hci_basic_led(struct led_trigger *trig, bool inc)
>>>>> +{
>>>>> +	struct hci_basic_led_trigger *htrig;
>>>>> +	int cnt;
>>>>> +
>>>>> +	if (!trig)
>>>>> +		return;
>>>>> +
>>>>> +	htrig = to_hci_basic_led_trigger(trig);
>>>>> +
>>>>> +	if (inc)
>>>>> +		cnt = atomic_inc_return(&htrig->cnt);
>>>>> +	else
>>>>> +		cnt = atomic_dec_return(&htrig->cnt);
>>>>> +
>>>>> +	WARN_ON(cnt < 0);
>>>>> +
>>>>> +	led_trigger_event(trig, cnt ? LED_FULL : LED_OFF);
>>>>> +}
>>>> 
>>>> Explain to me the reason for the counter? The Bluetooth subsystem is already guaranteeing the state of HCI_UP to be 100% synchronized. If have one case where it gets set and another where it gets unset. Meaning your counter will do 0 -> 1 and then 1 -> 0.
>>>> 
>>> Indeed the counter is not needed in this particular case.
>>> My intention was to make this basic led class more generic so that it can be re-used for triggers
>>> (to be added later) based on use cases like "at least one connection" where counting may be needed.
>> 
>> Actually it isn't needed there either. We have a hci_conn_hash that does the counting for you already. You pretty much just have to sum it up correctly and you know the correct state of your led.
>> 
> OK. We just might have one remaining issue: When the trigger gets activated, how to find out whether hci currently is powered or not?
> We can use the counter (as is) or we could store hdev in our trigger class and use hdev_is_powered(hdev) from within the activate callback.
> Would you prefer the latter approach?

Do not intermix some functions that are designed for the mgmt with low-level transport related handling. If you are adding the LED trigger where you currently do, then this is test_bit(HCI_UP, &hdev->flags).

Using hdev_is_powered would interact with the HCI_AUTO_OFF capability. And that is really not what you want. You better map it simple to the low-level transport that is HCI_UP flag. If you want to map it to the mgmt view of powered controllers, then that is a lot of more work. Especially since you then have to also deal with configured vs unconfigured controllers. And we also have raw mode and HCI User Channel which do not show in mgmt API.

Regards

Marcel


^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH v2] bluetooth: core: RfC - add basic LED triggers
  2016-01-05  7:03         ` Marcel Holtmann
@ 2016-01-05 20:12           ` Heiner Kallweit
  0 siblings, 0 replies; 7+ messages in thread
From: Heiner Kallweit @ 2016-01-05 20:12 UTC (permalink / raw)
  To: Marcel Holtmann; +Cc: Johan Hedberg, BlueZ development

Am 05.01.2016 um 08:03 schrieb Marcel Holtmann:
> Hi Heiner,
> 
>>>>>> In the bluetooth subsystem I miss the option to use LEDs to indicate
>>>>>> the status. This patch is based on the LED triggers in mac80211 and
>>>>>> adds two basic LED triggers (powered up, device connected) plus
>>>>>> a config option to enable the BT LED triggers.
>>>>>>
>>>>>> It works fine on my system however I don't know the bluetooth stack
>>>>>> in too much detail and therefore appreciate any comment.
>>>>>>
>>>>>> Signed-off-by: Heiner Kallweit <hkallweit1@gmail.com>
>>>>>> ---
>>>>>> v2:
>>>>>> - completely rewrote the code to be as generic as possible and minimize
>>>>>> the changes to struct hci_dev
>>>>>> - make the details of the LED triggers opaque to struct hci_dev
>>>>>> - avoid using ifdef CONFIG_BT_LEDS in hci_core.h
>>>>>> - start with the radio trigger only
>>>>>> - factor out the generic basic led trigger code to facilitate adding
>>>>>> further basic LED triggers
>>>>>> - fix indentation in Kconfig
>>>>>> - rename led.[ch] to leds.[ch]
>>>>>> ---
>>>>>> include/net/bluetooth/hci_core.h |  3 ++
>>>>>> net/bluetooth/Kconfig            |  9 ++++
>>>>>> net/bluetooth/Makefile           |  1 +
>>>>>> net/bluetooth/hci_core.c         |  8 ++++
>>>>>> net/bluetooth/leds.c             | 99 ++++++++++++++++++++++++++++++++++++++++
>>>>>> net/bluetooth/leds.h             | 17 +++++++
>>>>>> 6 files changed, 137 insertions(+)
>>>>>> create mode 100644 net/bluetooth/leds.c
>>>>>> create mode 100644 net/bluetooth/leds.h
>>>>>>
>>>>>> diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h
>>>>>> index c95e032..4ffec8b 100644
>>>>>> --- a/include/net/bluetooth/hci_core.h
>>>>>> +++ b/include/net/bluetooth/hci_core.h
>>>>>> @@ -25,6 +25,7 @@
>>>>>> #ifndef __HCI_CORE_H
>>>>>> #define __HCI_CORE_H
>>>>>>
>>>>>> +#include <linux/leds.h>
>>>>>> #include <net/bluetooth/hci.h>
>>>>>> #include <net/bluetooth/hci_sock.h>
>>>>>>
>>>>>> @@ -395,6 +396,8 @@ struct hci_dev {
>>>>>> 	struct delayed_work	rpa_expired;
>>>>>> 	bdaddr_t		rpa;
>>>>>>
>>>>>> +	struct led_trigger *radio_led;
>>>>>> +
>>>>>
>>>>> I wonder if this one might not be better named power_led instead.
>>>>>
>>>> OK
>>>>
>>>>>> 	int (*open)(struct hci_dev *hdev);
>>>>>> 	int (*close)(struct hci_dev *hdev);
>>>>>> 	int (*flush)(struct hci_dev *hdev);
>>>>>> diff --git a/net/bluetooth/Kconfig b/net/bluetooth/Kconfig
>>>>>> index 95d1a66..06c31b9 100644
>>>>>> --- a/net/bluetooth/Kconfig
>>>>>> +++ b/net/bluetooth/Kconfig
>>>>>> @@ -69,6 +69,15 @@ config BT_6LOWPAN
>>>>>> 	help
>>>>>> 	  IPv6 compression over Bluetooth Low Energy.
>>>>>>
>>>>>> +config BT_LEDS
>>>>>> +	bool "Enable LED triggers"
>>>>>> +	depends on BT
>>>>>> +	depends on LEDS_CLASS
>>>>>> +	select LEDS_TRIGGERS
>>>>>> +	help
>>>>>> +	  This option selects a few LED triggers for different
>>>>>> +	  Bluetooth events.
>>>>>> +
>>>>>> config BT_SELFTEST
>>>>>> 	bool "Bluetooth self testing support"
>>>>>> 	depends on BT && DEBUG_KERNEL
>>>>>> diff --git a/net/bluetooth/Makefile b/net/bluetooth/Makefile
>>>>>> index 2b15ae8..b3ff12e 100644
>>>>>> --- a/net/bluetooth/Makefile
>>>>>> +++ b/net/bluetooth/Makefile
>>>>>> @@ -17,6 +17,7 @@ bluetooth-y := af_bluetooth.o hci_core.o hci_conn.o hci_event.o mgmt.o \
>>>>>>
>>>>>> bluetooth-$(CONFIG_BT_BREDR) += sco.o
>>>>>> bluetooth-$(CONFIG_BT_HS) += a2mp.o amp.o
>>>>>> +bluetooth-$(CONFIG_BT_LEDS) += leds.o
>>>>>> bluetooth-$(CONFIG_BT_DEBUGFS) += hci_debugfs.o
>>>>>> bluetooth-$(CONFIG_BT_SELFTEST) += selftest.o
>>>>>>
>>>>>> diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c
>>>>>> index 47bcef7..f472b1d 100644
>>>>>> --- a/net/bluetooth/hci_core.c
>>>>>> +++ b/net/bluetooth/hci_core.c
>>>>>> @@ -40,6 +40,7 @@
>>>>>> #include "hci_request.h"
>>>>>> #include "hci_debugfs.h"
>>>>>> #include "smp.h"
>>>>>> +#include "leds.h"
>>>>>>
>>>>>> static void hci_rx_work(struct work_struct *work);
>>>>>> static void hci_cmd_work(struct work_struct *work);
>>>>>> @@ -1395,6 +1396,7 @@ static int hci_dev_do_open(struct hci_dev *hdev)
>>>>>> 		hci_dev_set_flag(hdev, HCI_RPA_EXPIRED);
>>>>>> 		set_bit(HCI_UP, &hdev->flags);
>>>>>> 		hci_sock_dev_event(hdev, HCI_DEV_UP);
>>>>>> +		hci_led_radio(hdev, true);
>>>>>
>>>>> Lets name this hci_leds_* etc.
>>>>>
>>>>> And we might want to consider not using the word radio since we never really have used that word anywhere else in the subsystem.
>>>>>
>>>> Then let's use the word power here too (e.g. hci_leds_powered).
>>>
>>> I am not 100% sure that is the best name either, but it has at least better meaning than radio.
>>>
>> Hmm, maybe hci_leds_ispowered ?
> 
> That is more like a query that should return true/false. Maybe hci_leds_update_powered.
> 
OK

>>>>
>>>>>> 		if (!hci_dev_test_flag(hdev, HCI_SETUP) &&
>>>>>> 		    !hci_dev_test_flag(hdev, HCI_CONFIG) &&
>>>>>> 		    !hci_dev_test_flag(hdev, HCI_UNCONFIGURED) &&
>>>>>> @@ -1532,6 +1534,8 @@ int hci_dev_do_close(struct hci_dev *hdev)
>>>>>> 		return 0;
>>>>>> 	}
>>>>>>
>>>>>> +	hci_led_radio(hdev, false);
>>>>>> +
>>>>>> 	/* Flush RX and TX works */
>>>>>> 	flush_work(&hdev->tx_work);
>>>>>> 	flush_work(&hdev->rx_work);
>>>>>> @@ -3067,6 +3071,8 @@ int hci_register_dev(struct hci_dev *hdev)
>>>>>> 	if (error < 0)
>>>>>> 		goto err_wqueue;
>>>>>>
>>>>>> +	hci_led_init(hdev);
>>>>>> +
>>>>>> 	hdev->rfkill = rfkill_alloc(hdev->name, &hdev->dev,
>>>>>> 				    RFKILL_TYPE_BLUETOOTH, &hci_rfkill_ops,
>>>>>> 				    hdev);
>>>>>> @@ -3128,6 +3134,8 @@ void hci_unregister_dev(struct hci_dev *hdev)
>>>>>>
>>>>>> 	id = hdev->id;
>>>>>>
>>>>>> +	hci_led_exit(hdev);
>>>>>> +
>>>>>> 	write_lock(&hci_dev_list_lock);
>>>>>> 	list_del(&hdev->list);
>>>>>> 	write_unlock(&hci_dev_list_lock);
>>>>>> diff --git a/net/bluetooth/leds.c b/net/bluetooth/leds.c
>>>>>> new file mode 100644
>>>>>> index 0000000..377f643
>>>>>> --- /dev/null
>>>>>> +++ b/net/bluetooth/leds.c
>>>>>> @@ -0,0 +1,99 @@
>>>>>> +/*
>>>>>> + * Copyright 2015, Heiner Kallweit <hkallweit1@gmail.com>
>>>>>> + *
>>>>>> + * This program is free software; you can redistribute it and/or modify
>>>>>> + * it under the terms of the GNU General Public License version 2 as
>>>>>> + * published by the Free Software Foundation.
>>>>>> + */
>>>>>> +
>>>>>> +#include <net/bluetooth/bluetooth.h>
>>>>>> +#include <net/bluetooth/hci_core.h>
>>>>>> +
>>>>>> +#include "leds.h"
>>>>>> +
>>>>>> +struct hci_basic_led_trigger {
>>>>>> +	struct led_trigger	led_trigger;
>>>>>> +	atomic_t		cnt;
>>>>>> +};
>>>>>> +
>>>>>> +#define to_hci_basic_led_trigger(arg) container_of(arg, \
>>>>>> +					struct hci_basic_led_trigger, \
>>>>>> +					led_trigger)
>>>>>> +
>>>>>> +static void hci_basic_led(struct led_trigger *trig, bool inc)
>>>>>> +{
>>>>>> +	struct hci_basic_led_trigger *htrig;
>>>>>> +	int cnt;
>>>>>> +
>>>>>> +	if (!trig)
>>>>>> +		return;
>>>>>> +
>>>>>> +	htrig = to_hci_basic_led_trigger(trig);
>>>>>> +
>>>>>> +	if (inc)
>>>>>> +		cnt = atomic_inc_return(&htrig->cnt);
>>>>>> +	else
>>>>>> +		cnt = atomic_dec_return(&htrig->cnt);
>>>>>> +
>>>>>> +	WARN_ON(cnt < 0);
>>>>>> +
>>>>>> +	led_trigger_event(trig, cnt ? LED_FULL : LED_OFF);
>>>>>> +}
>>>>>
>>>>> Explain to me the reason for the counter? The Bluetooth subsystem is already guaranteeing the state of HCI_UP to be 100% synchronized. If have one case where it gets set and another where it gets unset. Meaning your counter will do 0 -> 1 and then 1 -> 0.
>>>>>
>>>> Indeed the counter is not needed in this particular case.
>>>> My intention was to make this basic led class more generic so that it can be re-used for triggers
>>>> (to be added later) based on use cases like "at least one connection" where counting may be needed.
>>>
>>> Actually it isn't needed there either. We have a hci_conn_hash that does the counting for you already. You pretty much just have to sum it up correctly and you know the correct state of your led.
>>>
>> OK. We just might have one remaining issue: When the trigger gets activated, how to find out whether hci currently is powered or not?
>> We can use the counter (as is) or we could store hdev in our trigger class and use hdev_is_powered(hdev) from within the activate callback.
>> Would you prefer the latter approach?
> 
> Do not intermix some functions that are designed for the mgmt with low-level transport related handling. If you are adding the LED trigger where you currently do, then this is test_bit(HCI_UP, &hdev->flags).
> 
> Using hdev_is_powered would interact with the HCI_AUTO_OFF capability. And that is really not what you want. You better map it simple to the low-level transport that is HCI_UP flag. If you want to map it to the mgmt view of powered controllers, then that is a lot of more work. Especially since you then have to also deal with configured vs unconfigured controllers. And we also have raw mode and HCI User Channel which do not show in mgmt API.
> 
Thanks for the explanation. Then I'll go with test_bit(HCI_UP, &hdev->flags) and will provide a v3 of the patch.

> Regards
> 
> Marcel
> 
Heiner 

^ permalink raw reply	[flat|nested] 7+ messages in thread

end of thread, other threads:[~2016-01-05 20:12 UTC | newest]

Thread overview: 7+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2016-01-01 16:23 [PATCH v2] bluetooth: core: RfC - add basic LED triggers Heiner Kallweit
2016-01-04 18:19 ` Marcel Holtmann
2016-01-04 21:24   ` Heiner Kallweit
2016-01-04 21:28     ` Marcel Holtmann
2016-01-05  6:50       ` Heiner Kallweit
2016-01-05  7:03         ` Marcel Holtmann
2016-01-05 20:12           ` Heiner Kallweit

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.