Re: [PATCH v3 10/18] HID: steelseries: Add settings poll infrastructure

From: Bastien Nocera

Date: Tue Mar 03 2026 - 06:07:21 EST


On Fri, 2026-02-27 at 18:50 -0500, Sriman Achanta wrote:
> Some headset settings (sidetone level, mic volume, etc.) are not
> reported spontaneously but must be explicitly requested from the
> device.
> Introduce a separate delayed work item (settings_work) for fetching
> these persistent settings, independent of the existing status work.
>
> Settings are requested once at probe time and again whenever the
> headset
> reconnects after being disconnected. Device info structs gain
> request_settings and parse_settings hooks for model-specific
> implementations. The SS_CAP_EXTERNAL_CONFIG capability flag marks
> devices whose writable controls can also be changed from the headset
> hardware directly; writable ALSA controls on such devices will be
> marked
> volatile.

The settings polling work and the addition of SS_CAP_EXTERNAL_CONFIG
for some headsets should probably be made in separate commits.

>
> The initial implementation adds the Arctis Nova 7 Gen2 audio settings
> request (0x00, 0x20).
>
> Signed-off-by: Sriman Achanta <srimanachanta@xxxxxxxxx>
> ---
>  drivers/hid/hid-steelseries.c | 37
> ++++++++++++++++++++++++++++++++++-
>  1 file changed, 36 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/hid/hid-steelseries.c b/drivers/hid/hid-
> steelseries.c
> index 8c6116d02f19..f2423c350154 100644
> --- a/drivers/hid/hid-steelseries.c
> +++ b/drivers/hid/hid-steelseries.c
> @@ -26,6 +26,7 @@
>  #define SS_CAP_MIC_MUTE BIT(2)
>  #define SS_CAP_BT_ENABLED BIT(3)
>  #define SS_CAP_BT_DEVICE_CONNECTED BIT(4)
> +#define SS_CAP_EXTERNAL_CONFIG BIT(5)
>  
>  #define SS_QUIRK_STATUS_SYNC_POLL BIT(0)
>  
> @@ -40,6 +41,9 @@ struct steelseries_device_info {
>  
>   int (*request_status)(struct hid_device *hdev);
>   void (*parse_status)(struct steelseries_device *sd, u8
> *data, int size);
> +
> + int (*request_settings)(struct hid_device *hdev);
> + void (*parse_settings)(struct steelseries_device *sd, u8
> *data, int size);
>  };
>  
>  struct steelseries_device {
> @@ -49,6 +53,7 @@ struct steelseries_device {
>   bool use_async_protocol;
>  
>   struct delayed_work status_work;
> + struct delayed_work settings_work;
>  
>   struct power_supply_desc battery_desc;
>   struct power_supply *battery;
> @@ -690,6 +695,14 @@ static void
> steelseries_arctis_nova_7_gen2_parse_status(struct steelseries_devic
>   }
>  }
>  
> +static int steelseries_arctis_nova_7_gen2_request_settings(struct
> hid_device *hdev)
> +{
> + const u8 data[] = { 0x00, 0x20 };
> +
> + return steelseries_send_output_report(hdev, data,
> sizeof(data));
> +}
> +
> +
>  static void steelseries_arctis_nova_pro_parse_status(struct
> steelseries_device *sd,
>        u8 *data, int
> size)
>  {
> @@ -791,9 +804,11 @@ static const struct steelseries_device_info
> arctis_nova_7_gen2_info = {
>   .sync_interface = 3,
>   .async_interface = 5,
>   .capabilities = SS_CAP_BATTERY | SS_CAP_CHATMIX |
> SS_CAP_MIC_MUTE |
> - SS_CAP_BT_ENABLED |
> SS_CAP_BT_DEVICE_CONNECTED,
> + SS_CAP_BT_ENABLED |
> SS_CAP_BT_DEVICE_CONNECTED |
> + SS_CAP_EXTERNAL_CONFIG,
>   .request_status = steelseries_arctis_nova_request_status,
>   .parse_status = steelseries_arctis_nova_7_gen2_parse_status,
> + .request_settings =
> steelseries_arctis_nova_7_gen2_request_settings,
>  };
>  
>  static const struct steelseries_device_info arctis_nova_pro_info = {
> @@ -901,6 +916,15 @@ static void
> steelseries_status_timer_work_handler(struct work_struct *work)
>   spin_unlock_irqrestore(&sd->lock, flags);
>  }
>  
> +static void steelseries_settings_work_handler(struct work_struct
> *work)
> +{
> + struct steelseries_device *sd = container_of(
> + work, struct steelseries_device,
> settings_work.work);
> +
> + if (sd->info->request_settings)
> + sd->info->request_settings(sd->hdev);
> +}
> +
>  static int steelseries_battery_register(struct steelseries_device
> *sd)
>  {
>   static atomic_t battery_no = ATOMIC_INIT(0);
> @@ -1185,6 +1209,9 @@ static int steelseries_raw_event(struct
> hid_device *hdev,
>  
>   sd->info->parse_status(sd, data, size);
>  
> + if (sd->info->parse_settings)
> + sd->info->parse_settings(sd, data, size);
> +
>   if (sd->headset_connected != old_connected) {
>   hid_dbg(hdev,
>   "Connected status changed from %sconnected
> to %sconnected\n",
> @@ -1194,6 +1221,9 @@ static int steelseries_raw_event(struct
> hid_device *hdev,
>   if (sd->headset_connected && !old_connected &&
>       sd->use_async_protocol && is_async_interface) {
>   schedule_delayed_work(&sd->status_work, 0);
> + if (sd->info->request_settings)
> + schedule_delayed_work(&sd-
> >settings_work,
> +      
> msecs_to_jiffies(10));
>   }
>  
>   if (sd->battery) {
> @@ -1329,7 +1359,11 @@ static int steelseries_probe(struct hid_device
> *hdev,
>  #endif
>  
>   INIT_DELAYED_WORK(&sd->status_work,
> steelseries_status_timer_work_handler);
> + INIT_DELAYED_WORK(&sd->settings_work,
> steelseries_settings_work_handler);
> +
>   schedule_delayed_work(&sd->status_work,
> msecs_to_jiffies(100));
> + if (info->request_settings)
> + schedule_delayed_work(&sd->settings_work,
> msecs_to_jiffies(200));
>  
>   return 0;
>   }
> @@ -1415,6 +1449,7 @@ static void steelseries_remove(struct
> hid_device *hdev)
>   spin_unlock_irqrestore(&sd->lock, flags);
>  
>   cancel_delayed_work_sync(&sd->status_work);
> + cancel_delayed_work_sync(&sd->settings_work);
>   }
>  
>   hid_hw_close(hdev);