Bluetooth: Add support connectable advertising setting
authorMarcel Holtmann <marcel@holtmann.org>
Fri, 13 Mar 2015 05:30:58 +0000 (22:30 -0700)
committerJohan Hedberg <johan.hedberg@intel.com>
Fri, 13 Mar 2015 10:07:54 +0000 (12:07 +0200)
The patch adds a second advertising setting that allows switching of the
controller into connectable mode independent of the global connectable
setting.

Signed-off-by: Marcel Holtmann <marcel@holtmann.org>
Signed-off-by: Johan Hedberg <johan.hedberg@intel.com>
include/net/bluetooth/hci.h
net/bluetooth/mgmt.c

index f76f45ae76c3206b28af51366180f87ae79738bf..0f3413b285a5c487e7aa0407a65eff09ec325a76 100644 (file)
@@ -217,6 +217,7 @@ enum {
        HCI_HS_ENABLED,
        HCI_LE_ENABLED,
        HCI_ADVERTISING,
+       HCI_ADVERTISING_CONNECTABLE,
        HCI_CONNECTABLE,
        HCI_DISCOVERABLE,
        HCI_LIMITED_DISCOVERABLE,
index 025f29bf1f1a71b0f1097e7b43738f2c8c71a212..0761f2e0fefae6c3521b76b00e1f349c2050da59 100644 (file)
@@ -1110,7 +1110,10 @@ static void enable_advertising(struct hci_request *req)
         */
        clear_bit(HCI_LE_ADV, &hdev->dev_flags);
 
-       connectable = get_connectable(hdev);
+       if (test_bit(HCI_ADVERTISING_CONNECTABLE, &hdev->dev_flags))
+               connectable = true;
+       else
+               connectable = get_connectable(hdev);
 
        /* Set require_privacy to true only when non-connectable
         * advertising is used. In that case it is fine to use a
@@ -4430,7 +4433,7 @@ static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,
        struct mgmt_mode *cp = data;
        struct mgmt_pending_cmd *cmd;
        struct hci_request req;
-       u8 val, enabled, status;
+       u8 val, status;
        int err;
 
        BT_DBG("request for %s", hdev->name);
@@ -4440,29 +4443,42 @@ static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,
                return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
                                       status);
 
-       if (cp->val != 0x00 && cp->val != 0x01)
+       if (cp->val != 0x00 && cp->val != 0x01 && cp->val != 0x02)
                return mgmt_cmd_status(sk, hdev->id, MGMT_OP_SET_ADVERTISING,
                                       MGMT_STATUS_INVALID_PARAMS);
 
        hci_dev_lock(hdev);
 
        val = !!cp->val;
-       enabled = test_bit(HCI_ADVERTISING, &hdev->dev_flags);
 
        /* The following conditions are ones which mean that we should
         * not do any HCI communication but directly send a mgmt
         * response to user space (after toggling the flag if
         * necessary).
         */
-       if (!hdev_is_powered(hdev) || val == enabled ||
+       if (!hdev_is_powered(hdev) ||
+           (val == test_bit(HCI_ADVERTISING, &hdev->dev_flags) &&
+            (cp->val == 0x02) == test_bit(HCI_ADVERTISING_CONNECTABLE,
+                                          &hdev->dev_flags)) ||
            hci_conn_num(hdev, LE_LINK) > 0 ||
            (test_bit(HCI_LE_SCAN, &hdev->dev_flags) &&
             hdev->le_scan_type == LE_SCAN_ACTIVE)) {
-               bool changed = false;
+               bool changed;
 
-               if (val != test_bit(HCI_ADVERTISING, &hdev->dev_flags)) {
-                       change_bit(HCI_ADVERTISING, &hdev->dev_flags);
-                       changed = true;
+               if (cp->val) {
+                       changed = !test_and_set_bit(HCI_ADVERTISING,
+                                                   &hdev->dev_flags);
+                       if (cp->val == 0x02)
+                               set_bit(HCI_ADVERTISING_CONNECTABLE,
+                                       &hdev->dev_flags);
+                       else
+                               clear_bit(HCI_ADVERTISING_CONNECTABLE,
+                                         &hdev->dev_flags);
+               } else {
+                       changed = test_and_clear_bit(HCI_ADVERTISING,
+                                                    &hdev->dev_flags);
+                       clear_bit(HCI_ADVERTISING_CONNECTABLE,
+                                 &hdev->dev_flags);
                }
 
                err = send_settings_rsp(sk, MGMT_OP_SET_ADVERTISING, hdev);
@@ -4490,6 +4506,11 @@ static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,
 
        hci_req_init(&req, hdev);
 
+       if (cp->val == 0x02)
+               set_bit(HCI_ADVERTISING_CONNECTABLE, &hdev->dev_flags);
+       else
+               clear_bit(HCI_ADVERTISING_CONNECTABLE, &hdev->dev_flags);
+
        if (val)
                enable_advertising(&req);
        else