Received: by 2002:a05:6a10:22f:0:0:0:0 with SMTP id 15csp1108568pxk; Mon, 31 Aug 2020 09:58:57 -0700 (PDT) X-Google-Smtp-Source: ABdhPJyWFshx5H1fAIoFbNuChVtNPK7kzp87Wzl7h7l9sAa+EjG8snLyvdBIrme4hUiCa6DFpqHK X-Received: by 2002:a05:6402:1696:: with SMTP id a22mr2086979edv.126.1598893136939; Mon, 31 Aug 2020 09:58:56 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1598893136; cv=none; d=google.com; s=arc-20160816; b=R3BKT24BL38ueoN8wz2KR7HututQqum2/3IDjQ0EQHGs+QtBcMxbtKvP6wQwCrIAzl mGxQXaUnmuGjz6cZ87w/SQCpsMWM8UfUl2KFedRFbNBk79JAWpeqGAJ7E+OcneDS6+Bp qZQFMPC/elsFu56dsUMcWxEGHseFOvV6sbmVKklgdvxBEYKT5GBhYUFNMbdIrhqAWqVG 3OjcRs+ZSu1y8hMpGMWckPwMW6KNPupRKQC3TIUlSjtVPijhBjlQOGwVuVixDnDjH9ft 1XiNVM7225RMc8/7yInZKZUXrv1NWYUjLYriCkq8m1AXE8MpNZINoaPS374lLY0zYw05 iYDw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:cc:to:subject:message-id:date:from :in-reply-to:references:mime-version:dkim-signature; bh=wCvkvd1UYmEI0NU5tyMK6CLs+MKQ+O6ZyATWPxd+MyA=; b=KeDPq4HkAIfzlMt8nbezdt16u2f7Aqd4fVF0BvPdA7hf9dTXvaUbripZtIp5IT6A7+ 4DHFtb/2mLkpcV/QxOJxWCNr7i9HfpVNd7WmQxGOPYfM6gmsupmFrVrbx8tpxQBNA7rK 4SHJtXohqbCdgblieTG557Y5eqAr/tpvr8p57SW4lI8IYfgiqgmLfX3D1ukS8FzdjEf/ HANn9SHdAKAwAcFurMoxm3TnzKecZ3sCDZ9pqX1btIaXqg6IHweZZAqAqlal3kVabx4w 83BbYoR6+1JkRnyHlyi6u435IxiUvMCNa9Mxq4MbsXJq8k9CMhsgGgpVU4EqMRwudXp1 UJGQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=JLTnFjVM; spf=pass (google.com: domain of linux-bluetooth-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-bluetooth-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id k1si5774501ejp.718.2020.08.31.09.58.31; Mon, 31 Aug 2020 09:58:56 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-bluetooth-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=JLTnFjVM; spf=pass (google.com: domain of linux-bluetooth-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-bluetooth-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728306AbgHaQ5m (ORCPT + 99 others); Mon, 31 Aug 2020 12:57:42 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:40596 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727929AbgHaQ5k (ORCPT ); Mon, 31 Aug 2020 12:57:40 -0400 Received: from mail-oi1-x241.google.com (mail-oi1-x241.google.com [IPv6:2607:f8b0:4864:20::241]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 509CBC061573 for ; Mon, 31 Aug 2020 09:57:40 -0700 (PDT) Received: by mail-oi1-x241.google.com with SMTP id u126so219288oif.13 for ; Mon, 31 Aug 2020 09:57:40 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=mime-version:references:in-reply-to:from:date:message-id:subject:to :cc; bh=wCvkvd1UYmEI0NU5tyMK6CLs+MKQ+O6ZyATWPxd+MyA=; b=JLTnFjVMgRK40l5XPk+Ek5aPtTtB7SPZD2G4yGf46xwB3xuBSwLoYvbPy69wFuf90Y BlkFM3wVV7kUjn4fxk47oF+usix36AfZqsnXzyb197ZE37BdXg0jx6QiO1smWSFzKR/u McJ10UkAQyXgZ8z8SfAg/A2OKJJY/7Q3gQS8nO9HugCPU6WUUjt/Gu4qGJAr0T2XfDXh EQDA7aOySkeNwqeh4wF9FFEI2m/Hp+TtltoYTJ6pDfXcyIgL+L6US3HmdGkTtKX4+L8b QAb02+Ql6yyUrnxgGretHV7Vn1v21hhyfbW/TmnUxEh7gdk84Q3ly3CdWBvW5D9v4lp4 BeQQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:mime-version:references:in-reply-to:from:date :message-id:subject:to:cc; bh=wCvkvd1UYmEI0NU5tyMK6CLs+MKQ+O6ZyATWPxd+MyA=; b=pMxxW3CzZ02VXdIO86k6VcP0ZRi8a6dzfZsUmi/dOg9IBe6QyfCyCP+kCAc4ml18xT nqdvwFPWNKQLlJdopa6JWkrQakg9ZsWBZeI6rn+3o9fTlk/ZmXl3LDpwDwl5BjuXiJF/ PsF51ZpjpttiJM3ekQKTT6Wv6H315k8bSA0Irsfvfdgg7oEbx1iZB33WAbelwpMBavIR 0EFeZAJS2ag2cX5CQnCslHIxfjgfhlBfpP289TQLExxdH9Nr2TAkjt4X+LVgqu5RDDfC Od3jCD5cMrPYu8W59HwNTCrcsNLKURRx4vrUZwYxazC01khL2iNPIAf4Ts/6IJ2sk0cz 0VzA== X-Gm-Message-State: AOAM532IeqQSU31fsYqIQDaaHGsxe1S7cXAtiuOcpv3kWs7/WOBWp7Vu K0rxZLCyemw41Qv7Q16j+OBor/Ah/tuEYSTEcIw= X-Received: by 2002:aca:a9c9:: with SMTP id s192mr178702oie.152.1598893059439; Mon, 31 Aug 2020 09:57:39 -0700 (PDT) MIME-Version: 1.0 References: <20200829010210.871471-1-abhishekpandit@chromium.org> <20200828180157.Bluez.v3.3.I26efd89de3a70af1cd9775d457d0c10f4aafd4cb@changeid> In-Reply-To: <20200828180157.Bluez.v3.3.I26efd89de3a70af1cd9775d457d0c10f4aafd4cb@changeid> From: Luiz Augusto von Dentz Date: Mon, 31 Aug 2020 09:57:29 -0700 Message-ID: Subject: Re: [Bluez PATCH v3 3/3] policy: Reconnect audio on controller resume To: Abhishek Pandit-Subedi Cc: Marcel Holtmann , ChromeOS Bluetooth Upstreaming , "linux-bluetooth@vger.kernel.org" Content-Type: text/plain; charset="UTF-8" Sender: linux-bluetooth-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-bluetooth@vger.kernel.org Hi Abhishek, On Fri, Aug 28, 2020 at 6:02 PM Abhishek Pandit-Subedi wrote: > > During system suspend, all peer devices are disconnected. On resume, HID > devices will reconnect but audio devices stay disconnected. As a quality > of life improvement, mark audio devices that were disconnected due to > suspend and attempt to reconnect them when the controller resumes (after > a delay for better co-existence with Wi-Fi). > > --- > > Changes in v3: > - Refactored resume notification to use btd_adapter_driver > - Renamed ReconnectAudioDelay to ResumeDelay and set default to 2 > - Added A2DP_SINK_UUID to default reconnect list > > Changes in v2: > - Refactored to use policy instead of connecting directly in adapter > > plugins/policy.c | 80 +++++++++++++++++++++++++++++++++++++++--------- > src/adapter.c | 36 ++++++++++++++++++++++ > src/adapter.h | 2 ++ > src/main.c | 1 + > src/main.conf | 11 ++++++- > 5 files changed, 115 insertions(+), 15 deletions(-) > > diff --git a/plugins/policy.c b/plugins/policy.c > index de51e58b9..5ab65b991 100644 > --- a/plugins/policy.c > +++ b/plugins/policy.c > @@ -62,10 +62,12 @@ struct reconnect_data { > guint timer; > bool active; > unsigned int attempt; > + bool on_resume; > }; > > static const char *default_reconnect[] = { > - HSP_AG_UUID, HFP_AG_UUID, A2DP_SOURCE_UUID, NULL }; > + HSP_AG_UUID, HFP_AG_UUID, A2DP_SOURCE_UUID, > + A2DP_SINK_UUID, NULL }; Lets have this change in a separate patch. > static char **reconnect_uuids = NULL; > > static const size_t default_attempts = 7; > @@ -75,6 +77,9 @@ static const int default_intervals[] = { 1, 2, 4, 8, 16, 32, 64 }; > static int *reconnect_intervals = NULL; > static size_t reconnect_intervals_len = 0; > > +static const int default_resume_delay = 2; > +static int resume_delay; > + > static GSList *reconnects = NULL; > > static unsigned int service_id = 0; > @@ -711,6 +716,9 @@ static gboolean reconnect_timeout(gpointer data) > /* Mark the GSource as invalid */ > reconnect->timer = 0; > > + /* Mark any reconnect on resume as handled */ > + reconnect->on_resume = false; > + > err = btd_device_connect_services(reconnect->dev, reconnect->services); > if (err < 0) { > error("Reconnecting services failed: %s (%d)", > @@ -724,14 +732,17 @@ static gboolean reconnect_timeout(gpointer data) > return FALSE; > } > > -static void reconnect_set_timer(struct reconnect_data *reconnect) > +static void reconnect_set_timer(struct reconnect_data *reconnect, int timeout) > { > - static int timeout = 0; > + static int interval_timeout = 0; > > reconnect->active = true; > > if (reconnect->attempt < reconnect_intervals_len) > - timeout = reconnect_intervals[reconnect->attempt]; > + interval_timeout = reconnect_intervals[reconnect->attempt]; > + > + if (timeout < 0) > + timeout = interval_timeout; > > DBG("attempt %u/%zu %d seconds", reconnect->attempt + 1, > reconnect_attempts, timeout); > @@ -743,10 +754,14 @@ static void reconnect_set_timer(struct reconnect_data *reconnect) > static void disconnect_cb(struct btd_device *dev, uint8_t reason) > { > struct reconnect_data *reconnect; > + struct btd_service *service; > + struct policy_data *data; > > DBG("reason %u", reason); > > - if (reason != MGMT_DEV_DISCONN_TIMEOUT) > + /* Only attempt reconnect for the following reasons */ > + if (reason != MGMT_DEV_DISCONN_TIMEOUT && > + reason != MGMT_DEV_DISCONN_LOCAL_HOST_SUSPEND) We might want to rework this to have the reason just checked once. > return; > > reconnect = reconnect_find(dev); > @@ -755,10 +770,40 @@ static void disconnect_cb(struct btd_device *dev, uint8_t reason) > > reconnect_reset(reconnect); > > - DBG("Device %s identified for auto-reconnection", > - device_get_path(dev)); > + DBG("Device %s identified for auto-reconnection", device_get_path(dev)); > > - reconnect_set_timer(reconnect); > + switch(reason) > + { > + case MGMT_DEV_DISCONN_LOCAL_HOST_SUSPEND: > + if (btd_device_get_service(dev, A2DP_SINK_UUID)) { > + reconnect->on_resume = true; > + DBG("%s configured to reconnect on resume", > + device_get_path(dev)); I thought your idea was to set the timer here and then reset it on_resume so it would work even when the kernel does not indicate the resume with an event. > + } > + break; > + case MGMT_DEV_DISCONN_TIMEOUT: > + reconnect_set_timer(reconnect, -1); > + break; > + default: > + DBG("Developer error. Reason = %d", reason); > + break; > + } > +} > + > +static void policy_adapter_resume(struct btd_adapter *adapter, > + uint8_t wake_reason, const bdaddr_t *addr, > + const uint8_t addr_type) > +{ > + GSList *l; > + > + /* Check if any devices needed to be reconnected on resume */ > + for (l = reconnects; l; l = g_slist_next(l)) { > + struct reconnect_data *reconnect = l->data; We should be checking if the reconnect matches the adapter otherwise this could potentially call reconnect_set_timer multiple times for the same device if there are multiple adapters on the system. > + if (reconnect->on_resume) { > + reconnect_set_timer(reconnect, resume_delay); > + } > + } > } > > static void conn_fail_cb(struct btd_device *dev, uint8_t status) > @@ -786,14 +831,15 @@ static void conn_fail_cb(struct btd_device *dev, uint8_t status) > return; > } > > - reconnect_set_timer(reconnect); > + reconnect_set_timer(reconnect, -1); > } > > static int policy_adapter_probe(struct btd_adapter *adapter) > { > DBG(""); > > - btd_adapter_restore_powered(adapter); > + if (auto_enable) > + btd_adapter_restore_powered(adapter); > > return 0; > } > @@ -801,6 +847,7 @@ static int policy_adapter_probe(struct btd_adapter *adapter) > static struct btd_adapter_driver policy_driver = { > .name = "policy", > .probe = policy_adapter_probe, > + .resume = policy_adapter_resume, > }; > > static int policy_init(void) > @@ -854,14 +901,20 @@ static int policy_init(void) > auto_enable = g_key_file_get_boolean(conf, "Policy", "AutoEnable", > NULL); > > + resume_delay = g_key_file_get_integer( > + conf, "Policy", "ResumeDelay", &gerr); > + > + if (gerr) { > + g_clear_error(&gerr); > + resume_delay = default_resume_delay; > + } > done: > if (reconnect_uuids && reconnect_uuids[0] && reconnect_attempts) { > btd_add_disconnect_cb(disconnect_cb); > btd_add_conn_fail_cb(conn_fail_cb); > } > > - if (auto_enable) > - btd_register_adapter_driver(&policy_driver); > + btd_register_adapter_driver(&policy_driver); > > return 0; > } > @@ -882,8 +935,7 @@ static void policy_exit(void) > > btd_service_remove_state_cb(service_id); > > - if (auto_enable) > - btd_unregister_adapter_driver(&policy_driver); > + btd_unregister_adapter_driver(&policy_driver); > } > > BLUETOOTH_PLUGIN_DEFINE(policy, VERSION, BLUETOOTH_PLUGIN_PRIORITY_DEFAULT, > diff --git a/src/adapter.c b/src/adapter.c > index 1435e2bd7..bf355bfa7 100644 > --- a/src/adapter.c > +++ b/src/adapter.c > @@ -8776,6 +8776,37 @@ static void connected_callback(uint16_t index, uint16_t length, > eir_data_free(&eir_data); > } > > +static void controller_resume_notify(struct btd_adapter *adapter, > + const uint8_t wake_reason, > + const bdaddr_t *addr, > + const uint8_t addr_type) > +{ > + GSList *l; > + > + for (l = adapter->drivers; l; l = g_slist_next(l)) { > + struct btd_adapter_driver *driver = l->data; > + if (driver->resume) > + driver->resume(adapter, wake_reason, addr, addr_type); This should be in a separate patch, also not sure what is the reason to pass the address and addr_type on the resume callback? If the callback requires that it can query it internally. > + } > +} > + > +static void controller_resume_callback(uint16_t index, uint16_t length, > + const void *param, void *user_data) > +{ > + const struct mgmt_ev_controller_resume *ev = param; > + struct btd_adapter *adapter = user_data; > + > + if (length < sizeof(*ev)) { > + btd_error(adapter->dev_id, "Too small device resume event"); > + return; > + } > + > + info("Controller resume with wake event 0x%x", ev->wake_reason); > + > + controller_resume_notify(adapter, ev->wake_reason, &ev->addr.bdaddr, > + ev->addr.type); > +} > + > static void device_blocked_callback(uint16_t index, uint16_t length, > const void *param, void *user_data) > { > @@ -9399,6 +9430,11 @@ static void read_info_complete(uint8_t status, uint16_t length, > user_passkey_notify_callback, > adapter, NULL); > > + mgmt_register(adapter->mgmt, MGMT_EV_CONTROLLER_RESUME, > + adapter->dev_id, > + controller_resume_callback, > + adapter, NULL); > + > set_dev_class(adapter); > > set_name(adapter, btd_adapter_get_name(adapter)); > diff --git a/src/adapter.h b/src/adapter.h > index f8ac20261..b81ef6a04 100644 > --- a/src/adapter.h > +++ b/src/adapter.h > @@ -113,6 +113,8 @@ struct btd_adapter_driver { > const char *name; > int (*probe) (struct btd_adapter *adapter); > void (*remove) (struct btd_adapter *adapter); > + void (*resume)(struct btd_adapter *adapter, uint8_t wake_reason, > + const bdaddr_t *addr, const uint8_t addr_type); > }; > > typedef void (*service_auth_cb) (DBusError *derr, void *user_data); > diff --git a/src/main.c b/src/main.c > index b37c32948..038f867b5 100644 > --- a/src/main.c > +++ b/src/main.c > @@ -131,6 +131,7 @@ static const char *policy_options[] = { > "ReconnectAttempts", > "ReconnectIntervals", > "AutoEnable", > + "ResumeDelay", > NULL > }; > > diff --git a/src/main.conf b/src/main.conf > index 42f7e41c5..9f882e65a 100644 > --- a/src/main.conf > +++ b/src/main.conf > @@ -186,7 +186,7 @@ > # timeout). The policy plugin should contain a sane set of values by > # default, but this list can be overridden here. By setting the list to > # empty the reconnection feature gets disabled. > -#ReconnectUUIDs=00001112-0000-1000-8000-00805f9b34fb,0000111f-0000-1000-8000-00805f9b34fb,0000110a-0000-1000-8000-00805f9b34fb > +#ReconnectUUIDs=00001112-0000-1000-8000-00805f9b34fb,0000111f-0000-1000-8000-00805f9b34fb,0000110a-0000-1000-8000-00805f9b34fb,0000110b-0000-1000-8000-00805f9b34fb > > # ReconnectAttempts define the number of attempts to reconnect after a link > # lost. Setting the value to 0 disables reconnecting feature. > @@ -202,3 +202,12 @@ > # This includes adapters present on start as well as adapters that are plugged > # in later on. Defaults to 'false'. > #AutoEnable=false > + > +# Audio devices that were disconnected due to suspend will be reconnected on > +# resume. ResumeDelay determines the delay between when the controller > +# resumes from suspend and a connection attempt is made. A longer delay is > +# better for better co-existence with Wi-Fi. > +# The value is in seconds. > +# Default: 2 > +#ResumeDelay = 2 > + > -- > 2.28.0.402.g5ffc5be6b7-goog > -- Luiz Augusto von Dentz