/* * oxfw_command.c - a part of driver for OXFW970/971 based devices * * Copyright (c) 2014 Takashi Sakamoto * * Licensed under the terms of the GNU General Public License, version 2. */ #include "oxfw.h" int avc_stream_set_format(struct fw_unit *unit, enum avc_general_plug_dir dir, unsigned int pid, u8 *format, unsigned int len) { u8 *buf; int err; buf = kmalloc(len + 10, GFP_KERNEL); if (buf == NULL) return -ENOMEM; buf[0] = 0x00; /* CONTROL */ buf[1] = 0xff; /* UNIT */ buf[2] = 0xbf; /* EXTENDED STREAM FORMAT INFORMATION */ buf[3] = 0xc0; /* SINGLE subfunction */ buf[4] = dir; /* Plug Direction */ buf[5] = 0x00; /* UNIT */ buf[6] = 0x00; /* PCR (Isochronous Plug) */ buf[7] = 0xff & pid; /* Plug ID */ buf[8] = 0xff; /* Padding */ buf[9] = 0xff; /* Support status in response */ memcpy(buf + 10, format, len); /* do transaction and check buf[1-8] are the same against command */ err = fcp_avc_transaction(unit, buf, len + 10, buf, len + 10, BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) | BIT(6) | BIT(7) | BIT(8)); if ((err > 0) && (err < len + 10)) err = -EIO; else if (buf[0] == 0x08) /* NOT IMPLEMENTED */ err = -ENOSYS; else if (buf[0] == 0x0a) /* REJECTED */ err = -EINVAL; else err = 0; kfree(buf); return err; } int avc_stream_get_format(struct fw_unit *unit, enum avc_general_plug_dir dir, unsigned int pid, u8 *buf, unsigned int *len, unsigned int eid) { unsigned int subfunc; int err; if (eid == 0xff) subfunc = 0xc0; /* SINGLE */ else subfunc = 0xc1; /* LIST */ buf[0] = 0x01; /* STATUS */ buf[1] = 0xff; /* UNIT */ buf[2] = 0xbf; /* EXTENDED STREAM FORMAT INFORMATION */ buf[3] = subfunc; /* SINGLE or LIST */ buf[4] = dir; /* Plug Direction */ buf[5] = 0x00; /* Unit */ buf[6] = 0x00; /* PCR (Isochronous Plug) */ buf[7] = 0xff & pid; /* Plug ID */ buf[8] = 0xff; /* Padding */ buf[9] = 0xff; /* support status in response */ buf[10] = 0xff & eid; /* entry ID for LIST subfunction */ buf[11] = 0xff; /* padding */ /* do transaction and check buf[1-7] are the same against command */ err = fcp_avc_transaction(unit, buf, 12, buf, *len, BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5) | BIT(6) | BIT(7)); if ((err > 0) && (err < 10)) err = -EIO; else if (buf[0] == 0x08) /* NOT IMPLEMENTED */ err = -ENOSYS; else if (buf[0] == 0x0a) /* REJECTED */ err = -EINVAL; else if (buf[0] == 0x0b) /* IN TRANSITION */ err = -EAGAIN; /* LIST subfunction has entry ID */ else if ((subfunc == 0xc1) && (buf[10] != eid)) err = -EIO; if (err < 0) goto end; /* keep just stream format information */ if (subfunc == 0xc0) { memmove(buf, buf + 10, err - 10); *len = err - 10; } else { memmove(buf, buf + 11, err - 11); *len = err - 11; } err = 0; end: return err; } int avc_general_inquiry_sig_fmt(struct fw_unit *unit, unsigned int rate, enum avc_general_plug_dir dir, unsigned short pid) { unsigned int sfc; u8 *buf; int err; for (sfc = 0; sfc < CIP_SFC_COUNT; sfc++) { if (amdtp_rate_table[sfc] == rate) break; } if (sfc == CIP_SFC_COUNT) return -EINVAL; buf = kzalloc(8, GFP_KERNEL); if (buf == NULL) return -ENOMEM; buf[0] = 0x02; /* SPECIFIC INQUIRY */ buf[1] = 0xff; /* UNIT */ if (dir == AVC_GENERAL_PLUG_DIR_IN) buf[2] = 0x19; /* INPUT PLUG SIGNAL FORMAT */ else buf[2] = 0x18; /* OUTPUT PLUG SIGNAL FORMAT */ buf[3] = 0xff & pid; /* plug id */ buf[4] = 0x90; /* EOH_1, Form_1, FMT. AM824 */ buf[5] = 0x07 & sfc; /* FDF-hi. AM824, frequency */ buf[6] = 0xff; /* FDF-mid. AM824, SYT hi (not used) */ buf[7] = 0xff; /* FDF-low. AM824, SYT lo (not used) */ /* do transaction and check buf[1-5] are the same against command */ err = fcp_avc_transaction(unit, buf, 8, buf, 8, BIT(1) | BIT(2) | BIT(3) | BIT(4) | BIT(5)); if ((err > 0) && (err < 8)) err = -EIO; else if (buf[0] == 0x08) /* NOT IMPLEMENTED */ err = -ENOSYS; if (err < 0) goto end; err = 0; end: kfree(buf); return err; } select>
authorAnssi Hannula <anssi.hannula@iki.fi>2016-09-23 06:43:47 +0300
committerTakashi Iwai <tiwai@suse.de>2016-09-23 08:32:18 +0200
commiteb1a74b7bea17eea31915c4f76385cefe69d9795 (patch)
tree0fa235c5e35c02b8ab34e8dc6ef631764aa8a15b
parentdb68577966abc1aeae4ec597b3dcfa0d56e92041 (diff)
ALSA: usb-audio: Extend DragonFly dB scale quirk to cover other variants
The DragonFly quirk added in 42e3121d90f4 ("ALSA: usb-audio: Add a more accurate volume quirk for AudioQuest DragonFly") applies a custom dB map on the volume control when its range is reported as 0..50 (0 .. 0.2dB). However, there exists at least one other variant (hw v1.0c, as opposed to the tested v1.2) which reports a different non-sensical volume range (0..53) and the custom map is therefore not applied for that device. This results in all of the volume change appearing close to 100% on mixer UIs that utilize the dB TLV information. Add a fallback case where no dB TLV is reported at all if the control range is not 0..50 but still 0..N where N <= 1000 (3.9 dB). Also restrict the quirk to only apply to the volume control as there is also a mute control which would match the check otherwise. Fixes: 42e3121d90f4 ("ALSA: usb-audio: Add a more accurate volume quirk for AudioQuest DragonFly") Signed-off-by: Anssi Hannula <anssi.hannula@iki.fi> Reported-by: David W <regulars@d-dub.org.uk> Tested-by: David W <regulars@d-dub.org.uk> Cc: <stable@vger.kernel.org> Signed-off-by: Takashi Iwai <tiwai@suse.de>
Diffstat