sound fixes for 4.8-rc6
We've got quite a few fixes at this time, and all are stable patches. syzkaller strikes back again (episode 19 or so), and we had to plug some holes in ALSA core part (mostly timer). In addition, a couple of FireWire audio fixes for the invalid copy user calls in locks, and a few quirks for HD-audio and USB-audio as usual are included. -----BEGIN PGP SIGNATURE----- Version: GnuPG v2 iQIrBAABCAAVBQJX0u3IDhx0aXdhaUBzdXNlLmRlAAoJEGwxgFQ9KSmk8iYP/1cy t+Ap/5yKOGtdAoQDmB7yzXOh3Z1gbrjpSFx+/kOJnP/muHtJVcHGJkJpyMjshY0J 6qdQi6pq8HJakJWf9alZOcW260RYuXng9JwSt1xlo2z4zFjORNHyLeBA+FTwKQpl g1yyWoAyVHZC02alLe8YpNPQl1W06cHu4Tg3iTeZcMPwUEZ9QknUVCEj26Od61qR M2YK193F0Q32wsI26nyW/fOytE0avWc99ob0kJVC0GadduwkYODuiBDNtkEtur/h lvNPq7tOQVFOTTGdnMMTl1QhK8UDI3IxGnVAXWI3wqRMQolJtmimoJBHq4OXroK/ Nz+XOO4UAn+Tha7Lq+q8H8rjopB+EkaLfU+zq8+GourOh/aaVOL2EkClFA75qrxt w9F1HUMFICEgsn4vyaYxQx6hzL8YVEKXu4hCNpW36XxIc5RAnzEQmZvOp0xLLqfY OiDAnw1ciMKbtTNfgEOCubjN8owZBqrhV5b2ZGXSmpc2cVeKOn9p25FMVGuzqqMA cWMd/TvwU0yna/DxSRXCdRVKDX5RhcfP1qJD4R2YgzFWQHH6jzg8jy0nQ6D2TfIo 6leGUdIiDbQUO/lEezgew3BPB0fg6pzyFGisaU+v8p/Zg8C08Q2w0Pmv8pPT1BJt P1nEjhKWALhXrOBgvIPjpjRA4oXKkJBD92FS0eX+ =Jobo -----END PGP SIGNATURE----- Merge tag 'sound-4.8-rc6' of git://git.kernel.org/pub/scm/linux/kernel/git/tiwai/sound Pull sound fixes from Takashi Iwai: "We've got quite a few fixes at this time, and all are stable patches. syzkaller strikes back again (episode 19 or so), and we had to plug some holes in ALSA core part (mostly timer). In addition, a couple of FireWire audio fixes for the invalid copy user calls in locks, and a few quirks for HD-audio and USB-audio as usual are included" * tag 'sound-4.8-rc6' of git://git.kernel.org/pub/scm/linux/kernel/git/tiwai/sound: ALSA: rawmidi: Fix possible deadlock with virmidi registration ALSA: timer: Fix zero-division by continue of uninitialized instance ALSA: timer: fix NULL pointer dereference in read()/ioctl() race ALSA: fireworks: accessing to user space outside spinlock ALSA: firewire-tascam: accessing to user space outside spinlock ALSA: hda - Enable subwoofer on Dell Inspiron 7559 ALSA: hda - Add headset mic quirk for Dell Inspiron 5468 ALSA: usb-audio: Add sample rate inquiry quirk for B850V3 CP2114 ALSA: timer: fix NULL pointer dereference on memory allocation failure ALSA: timer: fix division by zero after SNDRV_TIMER_IOCTL_CONTINUE
This commit is contained in:
commit
067c2f472d
|
@ -1633,11 +1633,13 @@ static int snd_rawmidi_dev_register(struct snd_device *device)
|
||||||
return -EBUSY;
|
return -EBUSY;
|
||||||
}
|
}
|
||||||
list_add_tail(&rmidi->list, &snd_rawmidi_devices);
|
list_add_tail(&rmidi->list, &snd_rawmidi_devices);
|
||||||
|
mutex_unlock(®ister_mutex);
|
||||||
err = snd_register_device(SNDRV_DEVICE_TYPE_RAWMIDI,
|
err = snd_register_device(SNDRV_DEVICE_TYPE_RAWMIDI,
|
||||||
rmidi->card, rmidi->device,
|
rmidi->card, rmidi->device,
|
||||||
&snd_rawmidi_f_ops, rmidi, &rmidi->dev);
|
&snd_rawmidi_f_ops, rmidi, &rmidi->dev);
|
||||||
if (err < 0) {
|
if (err < 0) {
|
||||||
rmidi_err(rmidi, "unable to register\n");
|
rmidi_err(rmidi, "unable to register\n");
|
||||||
|
mutex_lock(®ister_mutex);
|
||||||
list_del(&rmidi->list);
|
list_del(&rmidi->list);
|
||||||
mutex_unlock(®ister_mutex);
|
mutex_unlock(®ister_mutex);
|
||||||
return err;
|
return err;
|
||||||
|
@ -1645,6 +1647,7 @@ static int snd_rawmidi_dev_register(struct snd_device *device)
|
||||||
if (rmidi->ops && rmidi->ops->dev_register &&
|
if (rmidi->ops && rmidi->ops->dev_register &&
|
||||||
(err = rmidi->ops->dev_register(rmidi)) < 0) {
|
(err = rmidi->ops->dev_register(rmidi)) < 0) {
|
||||||
snd_unregister_device(&rmidi->dev);
|
snd_unregister_device(&rmidi->dev);
|
||||||
|
mutex_lock(®ister_mutex);
|
||||||
list_del(&rmidi->list);
|
list_del(&rmidi->list);
|
||||||
mutex_unlock(®ister_mutex);
|
mutex_unlock(®ister_mutex);
|
||||||
return err;
|
return err;
|
||||||
|
@ -1677,7 +1680,6 @@ static int snd_rawmidi_dev_register(struct snd_device *device)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif /* CONFIG_SND_OSSEMUL */
|
#endif /* CONFIG_SND_OSSEMUL */
|
||||||
mutex_unlock(®ister_mutex);
|
|
||||||
sprintf(name, "midi%d", rmidi->device);
|
sprintf(name, "midi%d", rmidi->device);
|
||||||
entry = snd_info_create_card_entry(rmidi->card, name, rmidi->card->proc_root);
|
entry = snd_info_create_card_entry(rmidi->card, name, rmidi->card->proc_root);
|
||||||
if (entry) {
|
if (entry) {
|
||||||
|
|
|
@ -35,6 +35,9 @@
|
||||||
#include <sound/initval.h>
|
#include <sound/initval.h>
|
||||||
#include <linux/kmod.h>
|
#include <linux/kmod.h>
|
||||||
|
|
||||||
|
/* internal flags */
|
||||||
|
#define SNDRV_TIMER_IFLG_PAUSED 0x00010000
|
||||||
|
|
||||||
#if IS_ENABLED(CONFIG_SND_HRTIMER)
|
#if IS_ENABLED(CONFIG_SND_HRTIMER)
|
||||||
#define DEFAULT_TIMER_LIMIT 4
|
#define DEFAULT_TIMER_LIMIT 4
|
||||||
#else
|
#else
|
||||||
|
@ -294,8 +297,21 @@ int snd_timer_open(struct snd_timer_instance **ti,
|
||||||
get_device(&timer->card->card_dev);
|
get_device(&timer->card->card_dev);
|
||||||
timeri->slave_class = tid->dev_sclass;
|
timeri->slave_class = tid->dev_sclass;
|
||||||
timeri->slave_id = slave_id;
|
timeri->slave_id = slave_id;
|
||||||
if (list_empty(&timer->open_list_head) && timer->hw.open)
|
|
||||||
timer->hw.open(timer);
|
if (list_empty(&timer->open_list_head) && timer->hw.open) {
|
||||||
|
int err = timer->hw.open(timer);
|
||||||
|
if (err) {
|
||||||
|
kfree(timeri->owner);
|
||||||
|
kfree(timeri);
|
||||||
|
|
||||||
|
if (timer->card)
|
||||||
|
put_device(&timer->card->card_dev);
|
||||||
|
module_put(timer->module);
|
||||||
|
mutex_unlock(®ister_mutex);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
list_add_tail(&timeri->open_list, &timer->open_list_head);
|
list_add_tail(&timeri->open_list, &timer->open_list_head);
|
||||||
snd_timer_check_master(timeri);
|
snd_timer_check_master(timeri);
|
||||||
mutex_unlock(®ister_mutex);
|
mutex_unlock(®ister_mutex);
|
||||||
|
@ -526,6 +542,10 @@ static int snd_timer_stop1(struct snd_timer_instance *timeri, bool stop)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
timeri->flags &= ~(SNDRV_TIMER_IFLG_RUNNING | SNDRV_TIMER_IFLG_START);
|
timeri->flags &= ~(SNDRV_TIMER_IFLG_RUNNING | SNDRV_TIMER_IFLG_START);
|
||||||
|
if (stop)
|
||||||
|
timeri->flags &= ~SNDRV_TIMER_IFLG_PAUSED;
|
||||||
|
else
|
||||||
|
timeri->flags |= SNDRV_TIMER_IFLG_PAUSED;
|
||||||
snd_timer_notify1(timeri, stop ? SNDRV_TIMER_EVENT_STOP :
|
snd_timer_notify1(timeri, stop ? SNDRV_TIMER_EVENT_STOP :
|
||||||
SNDRV_TIMER_EVENT_CONTINUE);
|
SNDRV_TIMER_EVENT_CONTINUE);
|
||||||
unlock:
|
unlock:
|
||||||
|
@ -587,6 +607,10 @@ int snd_timer_stop(struct snd_timer_instance *timeri)
|
||||||
*/
|
*/
|
||||||
int snd_timer_continue(struct snd_timer_instance *timeri)
|
int snd_timer_continue(struct snd_timer_instance *timeri)
|
||||||
{
|
{
|
||||||
|
/* timer can continue only after pause */
|
||||||
|
if (!(timeri->flags & SNDRV_TIMER_IFLG_PAUSED))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
if (timeri->flags & SNDRV_TIMER_IFLG_SLAVE)
|
if (timeri->flags & SNDRV_TIMER_IFLG_SLAVE)
|
||||||
return snd_timer_start_slave(timeri, false);
|
return snd_timer_start_slave(timeri, false);
|
||||||
else
|
else
|
||||||
|
@ -813,6 +837,7 @@ int snd_timer_new(struct snd_card *card, char *id, struct snd_timer_id *tid,
|
||||||
timer->tmr_subdevice = tid->subdevice;
|
timer->tmr_subdevice = tid->subdevice;
|
||||||
if (id)
|
if (id)
|
||||||
strlcpy(timer->id, id, sizeof(timer->id));
|
strlcpy(timer->id, id, sizeof(timer->id));
|
||||||
|
timer->sticks = 1;
|
||||||
INIT_LIST_HEAD(&timer->device_list);
|
INIT_LIST_HEAD(&timer->device_list);
|
||||||
INIT_LIST_HEAD(&timer->open_list_head);
|
INIT_LIST_HEAD(&timer->open_list_head);
|
||||||
INIT_LIST_HEAD(&timer->active_list_head);
|
INIT_LIST_HEAD(&timer->active_list_head);
|
||||||
|
@ -1817,6 +1842,9 @@ static int snd_timer_user_continue(struct file *file)
|
||||||
tu = file->private_data;
|
tu = file->private_data;
|
||||||
if (!tu->timeri)
|
if (!tu->timeri)
|
||||||
return -EBADFD;
|
return -EBADFD;
|
||||||
|
/* start timer instead of continue if it's not used before */
|
||||||
|
if (!(tu->timeri->flags & SNDRV_TIMER_IFLG_PAUSED))
|
||||||
|
return snd_timer_user_start(file);
|
||||||
tu->timeri->lost = 0;
|
tu->timeri->lost = 0;
|
||||||
return (err = snd_timer_continue(tu->timeri)) < 0 ? err : 0;
|
return (err = snd_timer_continue(tu->timeri)) < 0 ? err : 0;
|
||||||
}
|
}
|
||||||
|
@ -1958,6 +1986,7 @@ static ssize_t snd_timer_user_read(struct file *file, char __user *buffer,
|
||||||
tu->qused--;
|
tu->qused--;
|
||||||
spin_unlock_irq(&tu->qlock);
|
spin_unlock_irq(&tu->qlock);
|
||||||
|
|
||||||
|
mutex_lock(&tu->ioctl_lock);
|
||||||
if (tu->tread) {
|
if (tu->tread) {
|
||||||
if (copy_to_user(buffer, &tu->tqueue[qhead],
|
if (copy_to_user(buffer, &tu->tqueue[qhead],
|
||||||
sizeof(struct snd_timer_tread)))
|
sizeof(struct snd_timer_tread)))
|
||||||
|
@ -1967,6 +1996,7 @@ static ssize_t snd_timer_user_read(struct file *file, char __user *buffer,
|
||||||
sizeof(struct snd_timer_read)))
|
sizeof(struct snd_timer_read)))
|
||||||
err = -EFAULT;
|
err = -EFAULT;
|
||||||
}
|
}
|
||||||
|
mutex_unlock(&tu->ioctl_lock);
|
||||||
|
|
||||||
spin_lock_irq(&tu->qlock);
|
spin_lock_irq(&tu->qlock);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
|
|
|
@ -108,7 +108,6 @@ struct snd_efw {
|
||||||
u8 *resp_buf;
|
u8 *resp_buf;
|
||||||
u8 *pull_ptr;
|
u8 *pull_ptr;
|
||||||
u8 *push_ptr;
|
u8 *push_ptr;
|
||||||
unsigned int resp_queues;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
int snd_efw_transaction_cmd(struct fw_unit *unit,
|
int snd_efw_transaction_cmd(struct fw_unit *unit,
|
||||||
|
|
|
@ -25,6 +25,7 @@ hwdep_read_resp_buf(struct snd_efw *efw, char __user *buf, long remained,
|
||||||
{
|
{
|
||||||
unsigned int length, till_end, type;
|
unsigned int length, till_end, type;
|
||||||
struct snd_efw_transaction *t;
|
struct snd_efw_transaction *t;
|
||||||
|
u8 *pull_ptr;
|
||||||
long count = 0;
|
long count = 0;
|
||||||
|
|
||||||
if (remained < sizeof(type) + sizeof(struct snd_efw_transaction))
|
if (remained < sizeof(type) + sizeof(struct snd_efw_transaction))
|
||||||
|
@ -38,8 +39,17 @@ hwdep_read_resp_buf(struct snd_efw *efw, char __user *buf, long remained,
|
||||||
buf += sizeof(type);
|
buf += sizeof(type);
|
||||||
|
|
||||||
/* write into buffer as many responses as possible */
|
/* write into buffer as many responses as possible */
|
||||||
while (efw->resp_queues > 0) {
|
spin_lock_irq(&efw->lock);
|
||||||
t = (struct snd_efw_transaction *)(efw->pull_ptr);
|
|
||||||
|
/*
|
||||||
|
* When another task reaches here during this task's access to user
|
||||||
|
* space, it picks up current position in buffer and can read the same
|
||||||
|
* series of responses.
|
||||||
|
*/
|
||||||
|
pull_ptr = efw->pull_ptr;
|
||||||
|
|
||||||
|
while (efw->push_ptr != pull_ptr) {
|
||||||
|
t = (struct snd_efw_transaction *)(pull_ptr);
|
||||||
length = be32_to_cpu(t->length) * sizeof(__be32);
|
length = be32_to_cpu(t->length) * sizeof(__be32);
|
||||||
|
|
||||||
/* confirm enough space for this response */
|
/* confirm enough space for this response */
|
||||||
|
@ -49,26 +59,39 @@ hwdep_read_resp_buf(struct snd_efw *efw, char __user *buf, long remained,
|
||||||
/* copy from ring buffer to user buffer */
|
/* copy from ring buffer to user buffer */
|
||||||
while (length > 0) {
|
while (length > 0) {
|
||||||
till_end = snd_efw_resp_buf_size -
|
till_end = snd_efw_resp_buf_size -
|
||||||
(unsigned int)(efw->pull_ptr - efw->resp_buf);
|
(unsigned int)(pull_ptr - efw->resp_buf);
|
||||||
till_end = min_t(unsigned int, length, till_end);
|
till_end = min_t(unsigned int, length, till_end);
|
||||||
|
|
||||||
if (copy_to_user(buf, efw->pull_ptr, till_end))
|
spin_unlock_irq(&efw->lock);
|
||||||
|
|
||||||
|
if (copy_to_user(buf, pull_ptr, till_end))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
efw->pull_ptr += till_end;
|
spin_lock_irq(&efw->lock);
|
||||||
if (efw->pull_ptr >= efw->resp_buf +
|
|
||||||
snd_efw_resp_buf_size)
|
pull_ptr += till_end;
|
||||||
efw->pull_ptr -= snd_efw_resp_buf_size;
|
if (pull_ptr >= efw->resp_buf + snd_efw_resp_buf_size)
|
||||||
|
pull_ptr -= snd_efw_resp_buf_size;
|
||||||
|
|
||||||
length -= till_end;
|
length -= till_end;
|
||||||
buf += till_end;
|
buf += till_end;
|
||||||
count += till_end;
|
count += till_end;
|
||||||
remained -= till_end;
|
remained -= till_end;
|
||||||
}
|
}
|
||||||
|
|
||||||
efw->resp_queues--;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* All of tasks can read from the buffer nearly simultaneously, but the
|
||||||
|
* last position for each task is different depending on the length of
|
||||||
|
* given buffer. Here, for simplicity, a position of buffer is set by
|
||||||
|
* the latest task. It's better for a listening application to allow one
|
||||||
|
* thread to read from the buffer. Unless, each task can read different
|
||||||
|
* sequence of responses depending on variation of buffer length.
|
||||||
|
*/
|
||||||
|
efw->pull_ptr = pull_ptr;
|
||||||
|
|
||||||
|
spin_unlock_irq(&efw->lock);
|
||||||
|
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -76,14 +99,17 @@ static long
|
||||||
hwdep_read_locked(struct snd_efw *efw, char __user *buf, long count,
|
hwdep_read_locked(struct snd_efw *efw, char __user *buf, long count,
|
||||||
loff_t *offset)
|
loff_t *offset)
|
||||||
{
|
{
|
||||||
union snd_firewire_event event;
|
union snd_firewire_event event = {
|
||||||
|
.lock_status.type = SNDRV_FIREWIRE_EVENT_LOCK_STATUS,
|
||||||
|
};
|
||||||
|
|
||||||
memset(&event, 0, sizeof(event));
|
spin_lock_irq(&efw->lock);
|
||||||
|
|
||||||
event.lock_status.type = SNDRV_FIREWIRE_EVENT_LOCK_STATUS;
|
|
||||||
event.lock_status.status = (efw->dev_lock_count > 0);
|
event.lock_status.status = (efw->dev_lock_count > 0);
|
||||||
efw->dev_lock_changed = false;
|
efw->dev_lock_changed = false;
|
||||||
|
|
||||||
|
spin_unlock_irq(&efw->lock);
|
||||||
|
|
||||||
count = min_t(long, count, sizeof(event.lock_status));
|
count = min_t(long, count, sizeof(event.lock_status));
|
||||||
|
|
||||||
if (copy_to_user(buf, &event, count))
|
if (copy_to_user(buf, &event, count))
|
||||||
|
@ -98,10 +124,15 @@ hwdep_read(struct snd_hwdep *hwdep, char __user *buf, long count,
|
||||||
{
|
{
|
||||||
struct snd_efw *efw = hwdep->private_data;
|
struct snd_efw *efw = hwdep->private_data;
|
||||||
DEFINE_WAIT(wait);
|
DEFINE_WAIT(wait);
|
||||||
|
bool dev_lock_changed;
|
||||||
|
bool queued;
|
||||||
|
|
||||||
spin_lock_irq(&efw->lock);
|
spin_lock_irq(&efw->lock);
|
||||||
|
|
||||||
while ((!efw->dev_lock_changed) && (efw->resp_queues == 0)) {
|
dev_lock_changed = efw->dev_lock_changed;
|
||||||
|
queued = efw->push_ptr != efw->pull_ptr;
|
||||||
|
|
||||||
|
while (!dev_lock_changed && !queued) {
|
||||||
prepare_to_wait(&efw->hwdep_wait, &wait, TASK_INTERRUPTIBLE);
|
prepare_to_wait(&efw->hwdep_wait, &wait, TASK_INTERRUPTIBLE);
|
||||||
spin_unlock_irq(&efw->lock);
|
spin_unlock_irq(&efw->lock);
|
||||||
schedule();
|
schedule();
|
||||||
|
@ -109,15 +140,17 @@ hwdep_read(struct snd_hwdep *hwdep, char __user *buf, long count,
|
||||||
if (signal_pending(current))
|
if (signal_pending(current))
|
||||||
return -ERESTARTSYS;
|
return -ERESTARTSYS;
|
||||||
spin_lock_irq(&efw->lock);
|
spin_lock_irq(&efw->lock);
|
||||||
|
dev_lock_changed = efw->dev_lock_changed;
|
||||||
|
queued = efw->push_ptr != efw->pull_ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (efw->dev_lock_changed)
|
|
||||||
count = hwdep_read_locked(efw, buf, count, offset);
|
|
||||||
else if (efw->resp_queues > 0)
|
|
||||||
count = hwdep_read_resp_buf(efw, buf, count, offset);
|
|
||||||
|
|
||||||
spin_unlock_irq(&efw->lock);
|
spin_unlock_irq(&efw->lock);
|
||||||
|
|
||||||
|
if (dev_lock_changed)
|
||||||
|
count = hwdep_read_locked(efw, buf, count, offset);
|
||||||
|
else if (queued)
|
||||||
|
count = hwdep_read_resp_buf(efw, buf, count, offset);
|
||||||
|
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -160,7 +193,7 @@ hwdep_poll(struct snd_hwdep *hwdep, struct file *file, poll_table *wait)
|
||||||
poll_wait(file, &efw->hwdep_wait, wait);
|
poll_wait(file, &efw->hwdep_wait, wait);
|
||||||
|
|
||||||
spin_lock_irq(&efw->lock);
|
spin_lock_irq(&efw->lock);
|
||||||
if (efw->dev_lock_changed || (efw->resp_queues > 0))
|
if (efw->dev_lock_changed || efw->pull_ptr != efw->push_ptr)
|
||||||
events = POLLIN | POLLRDNORM;
|
events = POLLIN | POLLRDNORM;
|
||||||
else
|
else
|
||||||
events = 0;
|
events = 0;
|
||||||
|
|
|
@ -188,8 +188,8 @@ proc_read_queues_state(struct snd_info_entry *entry,
|
||||||
else
|
else
|
||||||
consumed = (unsigned int)(efw->push_ptr - efw->pull_ptr);
|
consumed = (unsigned int)(efw->push_ptr - efw->pull_ptr);
|
||||||
|
|
||||||
snd_iprintf(buffer, "%d %d/%d\n",
|
snd_iprintf(buffer, "%d/%d\n",
|
||||||
efw->resp_queues, consumed, snd_efw_resp_buf_size);
|
consumed, snd_efw_resp_buf_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
|
@ -121,11 +121,11 @@ copy_resp_to_buf(struct snd_efw *efw, void *data, size_t length, int *rcode)
|
||||||
size_t capacity, till_end;
|
size_t capacity, till_end;
|
||||||
struct snd_efw_transaction *t;
|
struct snd_efw_transaction *t;
|
||||||
|
|
||||||
spin_lock_irq(&efw->lock);
|
|
||||||
|
|
||||||
t = (struct snd_efw_transaction *)data;
|
t = (struct snd_efw_transaction *)data;
|
||||||
length = min_t(size_t, be32_to_cpu(t->length) * sizeof(u32), length);
|
length = min_t(size_t, be32_to_cpu(t->length) * sizeof(u32), length);
|
||||||
|
|
||||||
|
spin_lock_irq(&efw->lock);
|
||||||
|
|
||||||
if (efw->push_ptr < efw->pull_ptr)
|
if (efw->push_ptr < efw->pull_ptr)
|
||||||
capacity = (unsigned int)(efw->pull_ptr - efw->push_ptr);
|
capacity = (unsigned int)(efw->pull_ptr - efw->push_ptr);
|
||||||
else
|
else
|
||||||
|
@ -155,7 +155,6 @@ copy_resp_to_buf(struct snd_efw *efw, void *data, size_t length, int *rcode)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* for hwdep */
|
/* for hwdep */
|
||||||
efw->resp_queues++;
|
|
||||||
wake_up(&efw->hwdep_wait);
|
wake_up(&efw->hwdep_wait);
|
||||||
|
|
||||||
*rcode = RCODE_COMPLETE;
|
*rcode = RCODE_COMPLETE;
|
||||||
|
|
|
@ -16,31 +16,14 @@
|
||||||
|
|
||||||
#include "tascam.h"
|
#include "tascam.h"
|
||||||
|
|
||||||
static long hwdep_read_locked(struct snd_tscm *tscm, char __user *buf,
|
|
||||||
long count)
|
|
||||||
{
|
|
||||||
union snd_firewire_event event;
|
|
||||||
|
|
||||||
memset(&event, 0, sizeof(event));
|
|
||||||
|
|
||||||
event.lock_status.type = SNDRV_FIREWIRE_EVENT_LOCK_STATUS;
|
|
||||||
event.lock_status.status = (tscm->dev_lock_count > 0);
|
|
||||||
tscm->dev_lock_changed = false;
|
|
||||||
|
|
||||||
count = min_t(long, count, sizeof(event.lock_status));
|
|
||||||
|
|
||||||
if (copy_to_user(buf, &event, count))
|
|
||||||
return -EFAULT;
|
|
||||||
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
|
|
||||||
static long hwdep_read(struct snd_hwdep *hwdep, char __user *buf, long count,
|
static long hwdep_read(struct snd_hwdep *hwdep, char __user *buf, long count,
|
||||||
loff_t *offset)
|
loff_t *offset)
|
||||||
{
|
{
|
||||||
struct snd_tscm *tscm = hwdep->private_data;
|
struct snd_tscm *tscm = hwdep->private_data;
|
||||||
DEFINE_WAIT(wait);
|
DEFINE_WAIT(wait);
|
||||||
union snd_firewire_event event;
|
union snd_firewire_event event = {
|
||||||
|
.lock_status.type = SNDRV_FIREWIRE_EVENT_LOCK_STATUS,
|
||||||
|
};
|
||||||
|
|
||||||
spin_lock_irq(&tscm->lock);
|
spin_lock_irq(&tscm->lock);
|
||||||
|
|
||||||
|
@ -54,10 +37,16 @@ static long hwdep_read(struct snd_hwdep *hwdep, char __user *buf, long count,
|
||||||
spin_lock_irq(&tscm->lock);
|
spin_lock_irq(&tscm->lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(&event, 0, sizeof(event));
|
event.lock_status.status = (tscm->dev_lock_count > 0);
|
||||||
count = hwdep_read_locked(tscm, buf, count);
|
tscm->dev_lock_changed = false;
|
||||||
|
|
||||||
spin_unlock_irq(&tscm->lock);
|
spin_unlock_irq(&tscm->lock);
|
||||||
|
|
||||||
|
count = min_t(long, count, sizeof(event.lock_status));
|
||||||
|
|
||||||
|
if (copy_to_user(buf, &event, count))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4855,6 +4855,7 @@ enum {
|
||||||
ALC221_FIXUP_HP_FRONT_MIC,
|
ALC221_FIXUP_HP_FRONT_MIC,
|
||||||
ALC292_FIXUP_TPT460,
|
ALC292_FIXUP_TPT460,
|
||||||
ALC298_FIXUP_SPK_VOLUME,
|
ALC298_FIXUP_SPK_VOLUME,
|
||||||
|
ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER,
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct hda_fixup alc269_fixups[] = {
|
static const struct hda_fixup alc269_fixups[] = {
|
||||||
|
@ -5516,6 +5517,15 @@ static const struct hda_fixup alc269_fixups[] = {
|
||||||
.chained = true,
|
.chained = true,
|
||||||
.chain_id = ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
|
.chain_id = ALC298_FIXUP_DELL1_MIC_NO_PRESENCE,
|
||||||
},
|
},
|
||||||
|
[ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER] = {
|
||||||
|
.type = HDA_FIXUP_PINS,
|
||||||
|
.v.pins = (const struct hda_pintbl[]) {
|
||||||
|
{ 0x1b, 0x90170151 },
|
||||||
|
{ }
|
||||||
|
},
|
||||||
|
.chained = true,
|
||||||
|
.chain_id = ALC255_FIXUP_DELL1_MIC_NO_PRESENCE
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct snd_pci_quirk alc269_fixup_tbl[] = {
|
static const struct snd_pci_quirk alc269_fixup_tbl[] = {
|
||||||
|
@ -5560,6 +5570,7 @@ static const struct snd_pci_quirk alc269_fixup_tbl[] = {
|
||||||
SND_PCI_QUIRK(0x1028, 0x06df, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
|
SND_PCI_QUIRK(0x1028, 0x06df, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
|
||||||
SND_PCI_QUIRK(0x1028, 0x06e0, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
|
SND_PCI_QUIRK(0x1028, 0x06e0, "Dell", ALC293_FIXUP_DISABLE_AAMIX_MULTIJACK),
|
||||||
SND_PCI_QUIRK(0x1028, 0x0704, "Dell XPS 13 9350", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
|
SND_PCI_QUIRK(0x1028, 0x0704, "Dell XPS 13 9350", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
|
||||||
|
SND_PCI_QUIRK(0x1028, 0x0706, "Dell Inspiron 7559", ALC256_FIXUP_DELL_INSPIRON_7559_SUBWOOFER),
|
||||||
SND_PCI_QUIRK(0x1028, 0x0725, "Dell Inspiron 3162", ALC255_FIXUP_DELL_SPK_NOISE),
|
SND_PCI_QUIRK(0x1028, 0x0725, "Dell Inspiron 3162", ALC255_FIXUP_DELL_SPK_NOISE),
|
||||||
SND_PCI_QUIRK(0x1028, 0x075b, "Dell XPS 13 9360", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
|
SND_PCI_QUIRK(0x1028, 0x075b, "Dell XPS 13 9360", ALC256_FIXUP_DELL_XPS_13_HEADPHONE_NOISE),
|
||||||
SND_PCI_QUIRK(0x1028, 0x075d, "Dell AIO", ALC298_FIXUP_SPK_VOLUME),
|
SND_PCI_QUIRK(0x1028, 0x075d, "Dell AIO", ALC298_FIXUP_SPK_VOLUME),
|
||||||
|
@ -5895,6 +5906,10 @@ static const struct snd_hda_pin_quirk alc269_pin_fixup_tbl[] = {
|
||||||
{0x12, 0x90a60170},
|
{0x12, 0x90a60170},
|
||||||
{0x14, 0x90170120},
|
{0x14, 0x90170120},
|
||||||
{0x21, 0x02211030}),
|
{0x21, 0x02211030}),
|
||||||
|
SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell Inspiron 5468", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
|
||||||
|
{0x12, 0x90a60180},
|
||||||
|
{0x14, 0x90170120},
|
||||||
|
{0x21, 0x02211030}),
|
||||||
SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
|
SND_HDA_PIN_QUIRK(0x10ec0256, 0x1028, "Dell", ALC255_FIXUP_DELL1_MIC_NO_PRESENCE,
|
||||||
ALC256_STANDARD_PINS),
|
ALC256_STANDARD_PINS),
|
||||||
SND_HDA_PIN_QUIRK(0x10ec0280, 0x103c, "HP", ALC280_FIXUP_HP_GPIO4,
|
SND_HDA_PIN_QUIRK(0x10ec0280, 0x103c, "HP", ALC280_FIXUP_HP_GPIO4,
|
||||||
|
|
|
@ -1141,6 +1141,7 @@ bool snd_usb_get_sample_rate_quirk(struct snd_usb_audio *chip)
|
||||||
case USB_ID(0x0556, 0x0014): /* Phoenix Audio TMX320VC */
|
case USB_ID(0x0556, 0x0014): /* Phoenix Audio TMX320VC */
|
||||||
case USB_ID(0x05A3, 0x9420): /* ELP HD USB Camera */
|
case USB_ID(0x05A3, 0x9420): /* ELP HD USB Camera */
|
||||||
case USB_ID(0x074D, 0x3553): /* Outlaw RR2150 (Micronas UAC3553B) */
|
case USB_ID(0x074D, 0x3553): /* Outlaw RR2150 (Micronas UAC3553B) */
|
||||||
|
case USB_ID(0x1901, 0x0191): /* GE B850V3 CP2114 audio interface */
|
||||||
case USB_ID(0x1de7, 0x0013): /* Phoenix Audio MT202exe */
|
case USB_ID(0x1de7, 0x0013): /* Phoenix Audio MT202exe */
|
||||||
case USB_ID(0x1de7, 0x0014): /* Phoenix Audio TMX320 */
|
case USB_ID(0x1de7, 0x0014): /* Phoenix Audio TMX320 */
|
||||||
case USB_ID(0x1de7, 0x0114): /* Phoenix Audio MT202pcs */
|
case USB_ID(0x1de7, 0x0114): /* Phoenix Audio MT202pcs */
|
||||||
|
|
Loading…
Reference in New Issue