From mboxrd@z Thu Jan 1 00:00:00 1970 Received: from mail-vs1-f42.google.com (mail-vs1-f42.google.com [209.85.217.42]) (using TLSv1.2 with cipher ECDHE-RSA-AES128-GCM-SHA256 (128/128 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 8BB073CF02C for ; Tue, 5 May 2026 20:01:55 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=209.85.217.42 ARC-Seal:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1778011317; cv=none; b=mvaF3FnjvEdhvUE2CXraPiIAz0ZromNewAzzCA9PElZSbMloDr8ZGaxyyZMJH9Sci9oVGmtfbkoJI2yBomREh6A05N1WHboNlJwMwcFFT11A7qNwDhakkX/GskGZ3hI4sXkJ+HjPvaaOELOt8wq5VjEBmWBCAN49ofKJpZpidNI= ARC-Message-Signature:i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1778011317; c=relaxed/simple; bh=0g2J4Odke1zqc6s4j4fTzWWywjj4kNZYQacuMZ7CG1A=; h=From:To:Subject:Date:Message-ID:MIME-Version; b=re4vlM2BWes2iRHOawJ3K97hgiwlsVQymA6rR250TshNbJ0w2myPptGFFpLHDRlj1bI70mLFPkhhNxY2H1K/ptZ4v7/H0bpIQSA4tc9S1DqZV4MCgkjq2aWVB/tYx3I7DSm2gTRWr7Ycjmct3PiG4P+zMC2ypXxcGEsGCuJ6uQ8= ARC-Authentication-Results:i=1; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=gmail.com; spf=pass smtp.mailfrom=gmail.com; dkim=pass (2048-bit key) header.d=gmail.com header.i=@gmail.com header.b=fP60uHZT; arc=none smtp.client-ip=209.85.217.42 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=gmail.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=gmail.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=gmail.com header.i=@gmail.com header.b="fP60uHZT" Received: by mail-vs1-f42.google.com with SMTP id ada2fe7eead31-62e94ae15f7so1140779137.3 for ; Tue, 05 May 2026 13:01:55 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20251104; t=1778011314; x=1778616114; darn=vger.kernel.org; h=content-transfer-encoding:mime-version:message-id:date:subject:to :from:from:to:cc:subject:date:message-id:reply-to; bh=gF6bO7xwpm+bdfz1Y3PaRnVsVTrYIOCmOsLB47EmnJ4=; b=fP60uHZTTSoE2wIkBVWqFvNLNqBcR2OGojuE91yywoML8mOV7QxYmAwyaDgp+HBQGs R/WkB3kaD7OJlCNE+68HWpMO/jD7UbjDK+Be1dqIyUy+ktuiKZYgNgYvqstTi2U8Mad6 94Jlba9CfMA0iK1a5QzEQlrtBjCyFDBupj7LJaLggO845qaqzn74VI1RKeG/S/TI0aoi J4xgYeukQ9qNHi/gjnmIdvoVhXCOGwHpYe1SzCh2SuxXk3ylXtGGncW9UOK5VHv59uk7 hXTshmOJzvtjJINUmrmpwinstWvnQiXXMDCIfTa432xH3we1cvrG24OXPHVfGmKHJcqd U/ew== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20251104; t=1778011314; x=1778616114; h=content-transfer-encoding:mime-version:message-id:date:subject:to :from:x-gm-gg:x-gm-message-state:from:to:cc:subject:date:message-id :reply-to; bh=gF6bO7xwpm+bdfz1Y3PaRnVsVTrYIOCmOsLB47EmnJ4=; b=XujBlc/n+vHIbiaRX5RjjKQwMEQ/qP+ig3J+qG+Z80liSaWzEFwLbhZ6oR/QN8N6UW U/457HEVYgSV529hNfix3rRBcqV9+Hvm5JRRBUw2c5XNTLcTjB4+8t/hMed2QGArgTET XfgX0bUJkC6xgvyaf9CyxlLu30l4AVsAuUn/siT2R03bM8wQuCynFIK41cV+nYtxIjmm OsM3yDXKiKKps/wlnqoUb5zIvgWKrjBEevqzeJITdMdb31LgszYsc5Ed/VS+6qXPwGXw yuBAlneFQEkrII2xc/mPSiw2OOxbKtMDdQzjHzXc9Guwul2/12L2XRgBgGQ0ilCIvI1P HCdA== X-Gm-Message-State: AOJu0Yw8D5oaX7wx0eQRxjd3SjSQkyaX3VCDl8wpBSPQnNZO/cjzy++Y 3miMIqwcaqx8CJboyGKezUY+d6Qrd6nQmbVMa0lb1AqdN17S7ANkLyky9dC22drs X-Gm-Gg: AeBDietDDmLLkhHIoyeQ4JWtimwTY06g0s7Gbsb7svzdXwuMnnTG5wlG75WRjPQct5n 0Ff91a0HPM6NQUBOGNK7WhOsZTAMF+A3CUWcgMEIA9RJDRFq5MYInOl7kqSN6kYwS+5qY4t0RAt bea2SykQiLw85Eoy+MZ5lGf/fphdfgacK3j01UiwhCaL4iJ4fp/AAVjea04Ca9Zbd7SNjMkDqYr YcrDCSpemIQUscAtt6oyU26YISMd3qop2yxwpiHG6BjRjGzAJZUNY3THjWLSihY8+jZhX+8ce54 NKNR3zZ4NJjHoMVSRwTnEUG0HOvK5BqqADspxi5XL16fx9DhHyT5TV0W1iruofFmmb0P47SVnSx yJpqf48SqYuL1OZwEHolWpbnBGQwmgDJWBl0jtOWj6ygQrADxzdrUTkVmXL6svhtb6wNG+oAxsJ oORMCYt7QbWVvZVB3BSTJJ+GRe4idYn9EypEocCS+EvPfojHO5C+qpGJ2KZuSuXPQZgpDDOM3mp qmsIHp30JS+Nti/NSwBrk2JdxTA X-Received: by 2002:a05:6102:5e8b:b0:62f:2f1f:599b with SMTP id ada2fe7eead31-630f8eb1656mr64189137.7.1778011314021; Tue, 05 May 2026 13:01:54 -0700 (PDT) Received: from lvondent-mobl5 ([72.188.211.115]) by smtp.gmail.com with ESMTPSA id a1e0cc1a2514c-95d0ac3217dsm4819391241.3.2026.05.05.13.01.52 for (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Tue, 05 May 2026 13:01:53 -0700 (PDT) From: Luiz Augusto von Dentz To: linux-bluetooth@vger.kernel.org Subject: [PATCH v1 1/2] Bluetooth: HCI: Add initial support for Short Connection Interval feature Date: Tue, 5 May 2026 16:01:46 -0400 Message-ID: <20260505200147.1325903-1-luiz.dentz@gmail.com> X-Mailer: git-send-email 2.53.0 Precedence: bulk X-Mailing-List: linux-bluetooth@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Transfer-Encoding: 8bit From: Luiz Augusto von Dentz This adds initial support for SCI related commands, command bits, event event mask bit and feature bits: Events: HCI_LE_Connection_Rate_Change(0x37) Commands: HCI_LE_Connection_Rate_Request(0x20a1) HCI_LE_Set_Default_Rate_Parameters(0x20a2) HCI_LE_Read_Minimum_Supported_Connection_Interval(0x20a3) Also update the init sequence to incorporte support for reading SCI groups and then setting the Default Rate Signed-off-by: Luiz Augusto von Dentz --- include/net/bluetooth/hci.h | 52 ++++++++++++++++++ include/net/bluetooth/hci_core.h | 13 +++++ net/bluetooth/hci_core.c | 3 +- net/bluetooth/hci_event.c | 72 +++++++++++++++++++++++++ net/bluetooth/hci_sync.c | 90 +++++++++++++++++++++++++++++++- 5 files changed, 228 insertions(+), 2 deletions(-) diff --git a/include/net/bluetooth/hci.h b/include/net/bluetooth/hci.h index 572b1c620c5d..848ec42de827 100644 --- a/include/net/bluetooth/hci.h +++ b/include/net/bluetooth/hci.h @@ -656,6 +656,7 @@ enum { #define HCI_LE_LL_EXT_FEATURE 0x80 #define HCI_LE_CS 0x40 #define HCI_LE_CS_HOST 0x80 +#define HCI_LE_SCI 0x01 /* Connection modes */ #define HCI_CM_ACTIVE 0x0000 @@ -2486,6 +2487,46 @@ struct hci_rp_le_cs_test { #define HCI_OP_LE_CS_TEST_END 0x2096 +#define HCI_OP_LE_CONN_RATE 0x20a1 +struct hci_cp_le_conn_rate { + __le16 handle; + __le16 interval_min; + __le16 interval_max; + __le16 subrate_min; + __le16 subrate_max; + __le16 max_latency; + __le16 cont_num; + __le16 supv_timeout; + __le16 min_ce_len; + __le16 max_ce_len; +} __packed; + +#define HCI_OP_LE_SET_DEF_RATE 0x20a2 +struct hci_cp_le_set_def_rate { + __le16 interval_min; + __le16 interval_max; + __le16 subrate_min; + __le16 subrate_max; + __le16 max_latency; + __le16 cont_num; + __le16 supv_timeout; + __le16 min_ce_len; + __le16 max_ce_len; +} __packed; + +#define HCI_OP_LE_READ_CONN_INTERVAL 0x20a3 +struct hci_le_conn_interval_group { + __le16 min; + __le16 max; + __le16 stride; +} __packed; + +struct hci_rp_le_read_conn_interval { + __u8 status; + __u8 num_grps; + struct hci_le_conn_interval_group grps[] __counted_by(num_grps); +} __packed; + /* ---- HCI Events ---- */ struct hci_ev_status { __u8 status; @@ -3300,6 +3341,17 @@ struct hci_evt_le_cs_test_end_complete { __u8 status; } __packed; +#define HCI_EVT_LE_CONN_RATE_CHANGE 0x37 +struct hci_evt_le_conn_rate_change { + __u8 status; + __le16 handle; + __le16 interval; + __le16 subrate; + __le16 latency; + __le16 cont_number; + __le16 supv_timeout; +} __packed; + #define HCI_EV_VENDOR 0xff /* Internal events generated by Bluetooth stack */ diff --git a/include/net/bluetooth/hci_core.h b/include/net/bluetooth/hci_core.h index aa600fbf9a53..cf4860c64f00 100644 --- a/include/net/bluetooth/hci_core.h +++ b/include/net/bluetooth/hci_core.h @@ -333,6 +333,14 @@ struct adv_monitor { #define HCI_ADV_MONITOR_EXT_NONE 1 #define HCI_ADV_MONITOR_EXT_MSFT 2 + +struct sci_group { + struct list_head list; + __u16 min; + __u16 max; + __u16 stride; +}; + #define HCI_MAX_SHORT_NAME_LENGTH 10 #define HCI_CONN_HANDLE_MAX 0x0eff @@ -572,6 +580,7 @@ struct hci_dev { struct list_head pend_le_reports; struct list_head blocked_keys; struct list_head local_codecs; + struct list_head sci_groups; struct hci_dev_stats stat; @@ -2082,6 +2091,10 @@ void hci_conn_del_sysfs(struct hci_conn *conn); #define mws_transport_config_capable(dev) (((dev)->commands[30] & 0x08) && \ (!hci_test_quirk((dev), HCI_QUIRK_BROKEN_MWS_TRANSPORT_CONFIG))) +/* Shorter Connection Intervals support */ +#define le_sci_capable(dev) \ + ((dev)->le_features[9] & HCI_LE_SCI) + /* ----- HCI protocols ----- */ #define HCI_PROTO_DEFER 0x01 diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c index c46c1236ebfa..95268440f48c 100644 --- a/net/bluetooth/hci_core.c +++ b/net/bluetooth/hci_core.c @@ -2543,8 +2543,9 @@ struct hci_dev *hci_alloc_dev_priv(int sizeof_priv) INIT_LIST_HEAD(&hdev->adv_instances); INIT_LIST_HEAD(&hdev->blocked_keys); INIT_LIST_HEAD(&hdev->monitored_devices); - INIT_LIST_HEAD(&hdev->local_codecs); + INIT_LIST_HEAD(&hdev->sci_groups); + INIT_WORK(&hdev->rx_work, hci_rx_work); INIT_WORK(&hdev->cmd_work, hci_cmd_work); INIT_WORK(&hdev->tx_work, hci_tx_work); diff --git a/net/bluetooth/hci_event.c b/net/bluetooth/hci_event.c index eea2f810aafa..ff24f4c7e05d 100644 --- a/net/bluetooth/hci_event.c +++ b/net/bluetooth/hci_event.c @@ -3957,6 +3957,61 @@ static u8 hci_cc_le_read_all_local_features(struct hci_dev *hdev, void *data, return rp->status; } +static void hci_sci_groups_clear(struct hci_dev *hdev) +{ + struct sci_group *grp, *tmp; + + list_for_each_entry_safe(grp, tmp, &hdev->sci_groups, list) { + list_del(&grp->list); + kfree(grp); + } +} + +static u8 hci_cc_le_read_conn_interval(struct hci_dev *hdev, void *data, + struct sk_buff *skb) +{ + struct hci_rp_le_read_conn_interval *rp = data; + __u8 i; + + bt_dev_dbg(hdev, "status 0x%2.2x", rp->status); + + if (rp->status || !rp->num_grps) + return rp->status; + + hci_dev_lock(hdev); + + /* Clear any existing SCI groups before adding new ones. */ + hci_sci_groups_clear(hdev); + + for (i = 0; i < rp->num_grps; i++) { + struct hci_le_conn_interval_group *grp; + struct sci_group *sgrp; + + /* Pull HCI event data for the current group. */ + grp = skb_pull_data(skb, sizeof(*grp)); + if (!grp) { + bt_dev_err(hdev, "invalid data length for SCI group"); + return rp->status; + } + + sgrp = kzalloc(sizeof(*sgrp), GFP_KERNEL); + if (!grp) { + bt_dev_err(hdev, "can't allocate memory for SCI group"); + return rp->status; + } + + sgrp->min = __le16_to_cpu(grp->min); + sgrp->max = __le16_to_cpu(grp->max); + sgrp->stride = __le16_to_cpu(grp->stride); + + list_add(&sgrp->list, &hdev->sci_groups); + } + + hci_dev_unlock(hdev); + + return rp->status; +} + static void hci_cs_le_create_big(struct hci_dev *hdev, u8 status) { bt_dev_dbg(hdev, "status 0x%2.2x", status); @@ -4239,6 +4294,10 @@ static const struct hci_cc { HCI_CC(HCI_OP_LE_READ_ALL_LOCAL_FEATURES, hci_cc_le_read_all_local_features, sizeof(struct hci_rp_le_read_all_local_features)), + HCI_CC_VL(HCI_OP_LE_READ_CONN_INTERVAL, + hci_cc_le_read_conn_interval, + sizeof(struct hci_rp_le_read_conn_interval), + HCI_MAX_EVENT_SIZE), }; static u8 hci_cc_func(struct hci_dev *hdev, const struct hci_cc *cc, @@ -7372,6 +7431,16 @@ static void hci_le_read_all_remote_features_evt(struct hci_dev *hdev, hci_dev_unlock(hdev); } +static void hci_le_conn_rate_change_evt(struct hci_dev *hdev, void *data, + struct sk_buff *skb) +{ + struct hci_evt_le_conn_rate_change *ev = data; + + bt_dev_dbg(hdev, "status 0x%2.2x", ev->status); + + /* TODO: Store rate to be used for next connection? */ +} + #define HCI_LE_EV_VL(_op, _func, _min_len, _max_len) \ [_op] = { \ .func = _func, \ @@ -7483,6 +7552,9 @@ static const struct hci_le_ev { sizeof(struct hci_evt_le_read_all_remote_features_complete), HCI_MAX_EVENT_SIZE), + /* [0x37 = HCI_EVT_LE_CONN_RATE_CHANGE] */ + HCI_LE_EV(HCI_EVT_LE_CONN_RATE_CHANGE, hci_le_conn_rate_change_evt, + sizeof(struct hci_evt_le_conn_rate_change)), }; static void hci_le_meta_evt(struct hci_dev *hdev, void *data, diff --git a/net/bluetooth/hci_sync.c b/net/bluetooth/hci_sync.c index fd3aacdea512..9e84e2912cde 100644 --- a/net/bluetooth/hci_sync.c +++ b/net/bluetooth/hci_sync.c @@ -4449,6 +4449,10 @@ static int hci_le_set_event_mask_sync(struct hci_dev *hdev) events[6] |= 0x02; /* LE CS Subevent Result Continue event */ events[6] |= 0x04; /* LE CS Test End Complete event */ } + + if (le_sci_capable(hdev)) + events[6] |= 0x20; /* LE Connection Rate Change */ + return __hci_cmd_sync_status(hdev, HCI_OP_LE_SET_EVENT_MASK, sizeof(events), events, HCI_CMD_TIMEOUT); } @@ -4611,9 +4615,16 @@ static int hci_le_set_host_features_sync(struct hci_dev *hdev) return err; } - if (le_cs_capable(hdev)) + if (le_cs_capable(hdev)) { /* Channel Sounding (Host Support) */ err = hci_le_set_host_feature_sync(hdev, 47, 0x01); + if (err) + return err; + } + + if (le_sci_capable(hdev)) + /* Short Connection Interval (Host Support) */ + err = hci_le_set_host_feature_sync(hdev, 73, 0x01); return err; } @@ -4896,11 +4907,88 @@ static int hci_le_set_default_phy_sync(struct hci_dev *hdev) sizeof(cp), &cp, HCI_CMD_TIMEOUT); } +/* Read Connection Interval if command is supported and SCI feature bit is + * marked as supported. + */ +static int hci_le_read_conn_interval_sync(struct hci_dev *hdev) +{ + if (!(hdev->commands[48] & BIT(7)) || + !(hdev->le_features[0] & HCI_LE_SCI)) + return 0; + + return __hci_cmd_sync_status(hdev, HCI_OP_LE_READ_CONN_INTERVAL, + 0, NULL, HCI_CMD_TIMEOUT); +} + +/* Set Default Connection Rate Parameters if command is supported, SCI feature + * bit is marked as supported and at least one of the supported SCI groups + * exists. + */ +static int hci_le_set_def_rate_sync(struct hci_dev *hdev) +{ + struct hci_cp_le_set_def_rate cp; + struct sci_group *grp, *tmp; + __u16 min, max; + + if (!(hdev->commands[48] & BIT(6)) || + !(hdev->le_features[0] & HCI_LE_SCI) || + list_empty(&hdev->sci_groups)) + return 0; + + memset(&cp, 0, sizeof(cp)); + + /* Iterate over the SCI groups and find the widest supported connection + * interval range to maximize compatibility with peer devices. + */ + list_for_each_entry_safe(grp, tmp, &hdev->sci_groups, list) { + if (!min == 0 || grp->min < min) + min = grp->min; + + if (!max == 0 || grp->max > max) + max = grp->max; + } + + cp.interval_min = cpu_to_le16(min); + cp.interval_max = cpu_to_le16(max); + + /* HOG 1.2 Table 7.4. Modes with recommended parameter values suggests + * subrate 1-4 for all modes so use that as default. + */ + cp.subrate_min = cpu_to_le16(0x0001); + cp.subrate_max = cpu_to_le16(0x0004); + + /* HIP 1.2 Table 7.5. Modes with recommended parameter values suggests + * max latency of 0 for all modes expect low power. + */ + cp.max_latency = 0x0000; + + /* HIP 1.2 Table 7.5. Modes with recommended parameter values suggests + * continuation number 1 for full range. + */ + cp.cont_num = cpu_to_le16(0x0001); + + /* HOG 1.2 Table 7.4. Modes with recommended parameter values states + * that link supervision timeout should be: Greater than or equal to + * (1 + Peripheral Latency) x Subrate max x Connection Interval max x 2. + */ + cp.supv_timeout = cpu_to_le16((1 + 0) * 0x0004 * max * 2 / 10); + + cp.min_ce_len = cpu_to_le16(min); + cp.max_ce_len = cpu_to_le16(max); + + return __hci_cmd_sync_status(hdev, HCI_OP_LE_SET_DEF_RATE, + sizeof(cp), &cp, HCI_CMD_TIMEOUT); +} + static const struct hci_init_stage le_init4[] = { /* HCI_OP_LE_WRITE_DEF_DATA_LEN */ HCI_INIT(hci_le_set_write_def_data_len_sync), /* HCI_OP_LE_SET_DEFAULT_PHY */ HCI_INIT(hci_le_set_default_phy_sync), + /* HCI_OP_LE_READ_CONN_INTERVAL */ + HCI_INIT(hci_le_read_conn_interval_sync), + /* HCI_OP_LE_SET_DEF_RATE */ + HCI_INIT(hci_le_set_def_rate_sync), {} }; -- 2.53.0