Linux bluetooth development
 help / color / mirror / Atom feed
From: Heiner Kallweit <hkallweit1@gmail.com>
To: Marcel Holtmann <marcel@holtmann.org>
Cc: Johan Hedberg <johan.hedberg@gmail.com>,
	BlueZ development <linux-bluetooth@vger.kernel.org>,
	Johannes Berg <johannes.berg@intel.com>
Subject: Re: [PATCH v2] bluetooth: core: RfC - add basic LED triggers
Date: Tue, 5 Jan 2016 07:50:13 +0100	[thread overview]
Message-ID: <568B67A5.9010502@gmail.com> (raw)
In-Reply-To: <46CEB4CF-16CD-401C-8A88-DB19B4AA1DB4@holtmann.org>

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

  reply	other threads:[~2016-01-05  6:50 UTC|newest]

Thread overview: 7+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
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 [this message]
2016-01-05  7:03         ` Marcel Holtmann
2016-01-05 20:12           ` Heiner Kallweit

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=568B67A5.9010502@gmail.com \
    --to=hkallweit1@gmail.com \
    --cc=johan.hedberg@gmail.com \
    --cc=johannes.berg@intel.com \
    --cc=linux-bluetooth@vger.kernel.org \
    --cc=marcel@holtmann.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox