From: Cryolitia PukNgae <cryolitia@uniontech.com>
Also improve debug logs for applied quirks
Signed-off-by: Cryolitia PukNgae <cryolitia@uniontech.com>
---
sound/usb/quirks.c | 82 +++++++++++++++++++++++++++++++++++++++++++++++++---
sound/usb/quirks.h | 3 ++
sound/usb/usbaudio.h | 1 +
3 files changed, 82 insertions(+), 4 deletions(-)
diff --git a/sound/usb/quirks.c b/sound/usb/quirks.c
index d736a4750356597bfb0f9d5ab01cdaeaac0f907c..94854f352b1702b491e1bf3c8b769f7088e03976 100644
--- a/sound/usb/quirks.c
+++ b/sound/usb/quirks.c
@@ -2446,6 +2446,62 @@ static const struct usb_audio_quirk_flags_table quirk_flags_table[] = {
{} /* terminator */
};
+static const char *const snd_usb_audio_quirk_flag_names[] = {
+ "get_sample_rate",
+ "share_media_device",
+ "align_transfer",
+ "tx_length",
+ "playback_first",
+ "skip_clock_selector",
+ "ignore_clock_source",
+ "itf_usb_dsd_dac",
+ "ctl_msg_delay",
+ "ctl_msg_delay_1m",
+ "ctl_msg_delay_5m",
+ "iface_delay",
+ "validate_rates",
+ "disable_autosuspend",
+ "ignore_ctl_error",
+ "dsd_raw",
+ "set_iface_first",
+ "generic_implicit_fb",
+ "skip_implicit_fb",
+ "iface_skip_close",
+ "force_iface_reset",
+ "fixed_rate",
+ "mic_res_16",
+ "mic_res_384",
+ "mixer_playback_min_mute",
+ "mixer_capture_min_mute",
+ NULL
+};
+
+const char *snd_usb_quirk_flag_find_name(unsigned long index)
+{
+ if (index >= ARRAY_SIZE(snd_usb_audio_quirk_flag_names))
+ return NULL;
+
+ return snd_usb_audio_quirk_flag_names[index];
+}
+
+u32 snd_usb_quirk_flags_from_name(char *name)
+{
+ u32 flag = 0;
+ u32 i;
+
+ if (!name || !*name)
+ return 0;
+
+ for (i = 0; snd_usb_audio_quirk_flag_names[i]; i++) {
+ if (strcmp(name, snd_usb_audio_quirk_flag_names[i]) == 0) {
+ flag = (1U << i);
+ break;
+ }
+ }
+
+ return flag;
+}
+
void snd_usb_init_quirk_flags(struct snd_usb_audio *chip)
{
const struct usb_audio_quirk_flags_table *p;
@@ -2454,10 +2510,28 @@ void snd_usb_init_quirk_flags(struct snd_usb_audio *chip)
if (chip->usb_id == p->id ||
(!USB_ID_PRODUCT(p->id) &&
USB_ID_VENDOR(chip->usb_id) == USB_ID_VENDOR(p->id))) {
- usb_audio_dbg(chip,
- "Set quirk_flags 0x%x for device %04x:%04x\n",
- p->flags, USB_ID_VENDOR(chip->usb_id),
- USB_ID_PRODUCT(chip->usb_id));
+ unsigned long flags = p->flags;
+ unsigned long bit;
+
+ for_each_set_bit(bit, &flags,
+ BYTES_TO_BITS(sizeof(p->flags))) {
+ const char *name =
+ snd_usb_audio_quirk_flag_names[bit];
+
+ if (name)
+ usb_audio_dbg(chip,
+ "Set quirk flag %s for device %04x:%04x\n",
+ name,
+ USB_ID_VENDOR(chip->usb_id),
+ USB_ID_PRODUCT(chip->usb_id));
+ else
+ usb_audio_warn(chip,
+ "Set unknown quirk flag 0x%lx for device %04x:%04x\n",
+ bit,
+ USB_ID_VENDOR(chip->usb_id),
+ USB_ID_PRODUCT(chip->usb_id));
+ }
+
chip->quirk_flags |= p->flags;
return;
}
diff --git a/sound/usb/quirks.h b/sound/usb/quirks.h
index f9bfd5ac7bab01717de3a76227482a128bf73165..bd5baf2b193a1985f3a0e52bf4a77ca741364769 100644
--- a/sound/usb/quirks.h
+++ b/sound/usb/quirks.h
@@ -50,4 +50,7 @@ void snd_usb_audioformat_attributes_quirk(struct snd_usb_audio *chip,
void snd_usb_init_quirk_flags(struct snd_usb_audio *chip);
+const char *snd_usb_quirk_flag_find_name(unsigned long flag);
+u32 snd_usb_quirk_flags_from_name(char *name);
+
#endif /* __USBAUDIO_QUIRKS_H */
diff --git a/sound/usb/usbaudio.h b/sound/usb/usbaudio.h
index 30b5102e3caed01eeb86d0075c41338104c58950..0a22cb4a02344b2dcf4009c560a759f2da25ca67 100644
--- a/sound/usb/usbaudio.h
+++ b/sound/usb/usbaudio.h
@@ -252,5 +252,6 @@ extern bool snd_usb_skip_validation;
#define QUIRK_FLAG_MIC_RES_384 (1U << 23)
#define QUIRK_FLAG_MIXER_PLAYBACK_MIN_MUTE (1U << 24)
#define QUIRK_FLAG_MIXER_CAPTURE_MIN_MUTE (1U << 25)
+/* Please also edit snd_usb_audio_quirk_flag_names */
#endif /* __USBAUDIO_H */
--
2.51.0
On Thu, 18 Sep 2025 11:24:30 +0200, Cryolitia PukNgae via B4 Relay wrote: > --- a/sound/usb/quirks.c > +++ b/sound/usb/quirks.c > @@ -2446,6 +2446,62 @@ static const struct usb_audio_quirk_flags_table quirk_flags_table[] = { > {} /* terminator */ > }; > > +static const char *const snd_usb_audio_quirk_flag_names[] = { > + "get_sample_rate", > + "share_media_device", I think it's worth to add a comment that this each entry corresponds to QUIRK_FLAG_XXX. Or another idea would be to define enums like: enum { QUIRK_TYPE_GET_SAMPLE_RATE, QUIRK_TYPE_SHARE_MEDIA_DEVICE, .... }; then redefine QUIRK_FLAG_* like: #define QUIRK_FLAG_GET_SAMPLE_RATE BIT_U32(QUIRK_TYPE_GET_SAMPLE_RATE) #define QUIRK_FLAG_SHARE_MEDIA_DEVICE BIT_U32(QUIRK_TYPE_SHARE_MEDIA_DEVICE) .... or #define QUIRK_FLAG(x) BIT_U32(QUIRK_TYPE_ ## x) and use like QUIRK_FLAG(GET_SAMPLE_RATE). With those changes, the above can be defined more safely like static const char *const snd_usb_audio_quirk_flag_names[] = { [QUIRK_TYPE_GET_SAMPLE_RATE] = "get_sample_rate", .... or even more drastically by defining some macro for each entry like: #define QUIRK_STRING_ENTRY(x) \ [QUIRK_TYPE_ ## x] = __stringify(x) and put like: static const char *const snd_usb_audio_quirk_flag_names[] = { QUIRK_STRING_ENTRY(GET_SAMPLE_RATE), .... }; In this case, it'll become upper letters, so the parse would need to deal with the case-insensitive comparison, though. > +u32 snd_usb_quirk_flags_from_name(char *name) Use const char *. > +{ > + u32 flag = 0; > + u32 i; The iterator can be simple int. > + if (!name || !*name) > + return 0; > + > + for (i = 0; snd_usb_audio_quirk_flag_names[i]; i++) { > + if (strcmp(name, snd_usb_audio_quirk_flag_names[i]) == 0) { > + flag = (1U << i); Use BIT_U32(i) > + break; We can return the value directly, so flag variable can be dropped. > void snd_usb_init_quirk_flags(struct snd_usb_audio *chip) > { > const struct usb_audio_quirk_flags_table *p; > @@ -2454,10 +2510,28 @@ void snd_usb_init_quirk_flags(struct snd_usb_audio *chip) > if (chip->usb_id == p->id || > (!USB_ID_PRODUCT(p->id) && > USB_ID_VENDOR(chip->usb_id) == USB_ID_VENDOR(p->id))) { > - usb_audio_dbg(chip, > - "Set quirk_flags 0x%x for device %04x:%04x\n", > - p->flags, USB_ID_VENDOR(chip->usb_id), > - USB_ID_PRODUCT(chip->usb_id)); > + unsigned long flags = p->flags; > + unsigned long bit; > + > + for_each_set_bit(bit, &flags, > + BYTES_TO_BITS(sizeof(p->flags))) { > + const char *name = > + snd_usb_audio_quirk_flag_names[bit]; > + > + if (name) > + usb_audio_dbg(chip, > + "Set quirk flag %s for device %04x:%04x\n", > + name, > + USB_ID_VENDOR(chip->usb_id), > + USB_ID_PRODUCT(chip->usb_id)); > + else > + usb_audio_warn(chip, > + "Set unknown quirk flag 0x%lx for device %04x:%04x\n", > + bit, > + USB_ID_VENDOR(chip->usb_id), > + USB_ID_PRODUCT(chip->usb_id)); > + } This could be better factored out as a function. thanks, Takashi
© 2016 - 2025 Red Hat, Inc.