2019-05-27 06:55:05 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2005-04-16 22:20:36 +00:00
|
|
|
/*
|
|
|
|
* ALSA driver for ICEnsemble ICE1712 (Envy24)
|
|
|
|
*
|
2007-10-15 07:50:19 +00:00
|
|
|
* Copyright (c) 2000 Jaroslav Kysela <perex@perex.cz>
|
2008-09-07 10:17:02 +00:00
|
|
|
*/
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
NOTES:
|
|
|
|
- spdif nonaudio consumer mode does not work (at least with my
|
|
|
|
Sony STR-DB830)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Changes:
|
|
|
|
*
|
|
|
|
* 2002.09.09 Takashi Iwai <tiwai@suse.de>
|
|
|
|
* split the code to several files. each low-level routine
|
|
|
|
* is stored in the local file and called from registration
|
|
|
|
* function from card_info struct.
|
|
|
|
*
|
|
|
|
* 2002.11.26 James Stafford <jstafford@ampltd.com>
|
|
|
|
* Added support for VT1724 (Envy24HT)
|
2008-09-07 10:17:02 +00:00
|
|
|
* I have left out support for 176.4 and 192 KHz for the moment.
|
2005-04-16 22:20:36 +00:00
|
|
|
* I also haven't done anything with the internal S/PDIF transmitter or the MPU-401
|
|
|
|
*
|
|
|
|
* 2003.02.20 Taksahi Iwai <tiwai@suse.de>
|
|
|
|
* Split vt1724 part to an independent driver.
|
|
|
|
* The GPIO is accessed through the callback functions now.
|
|
|
|
*
|
|
|
|
* 2004.03.31 Doug McLain <nostar@comcast.net>
|
|
|
|
* Added support for Event Electronics EZ8 card to hoontech.c.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/pci.h>
|
2006-03-22 09:53:19 +00:00
|
|
|
#include <linux/dma-mapping.h>
|
2005-04-16 22:20:36 +00:00
|
|
|
#include <linux/slab.h>
|
2011-07-15 17:13:37 +00:00
|
|
|
#include <linux/module.h>
|
2006-01-16 15:34:20 +00:00
|
|
|
#include <linux/mutex.h>
|
2006-03-28 09:56:48 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
#include <sound/core.h>
|
|
|
|
#include <sound/cs8427.h>
|
|
|
|
#include <sound/info.h>
|
|
|
|
#include <sound/initval.h>
|
2006-08-30 14:56:30 +00:00
|
|
|
#include <sound/tlv.h>
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
#include <sound/asoundef.h>
|
|
|
|
|
|
|
|
#include "ice1712.h"
|
|
|
|
|
|
|
|
/* lowlevel routines */
|
|
|
|
#include "delta.h"
|
|
|
|
#include "ews.h"
|
|
|
|
#include "hoontech.h"
|
|
|
|
|
2007-10-15 07:50:19 +00:00
|
|
|
MODULE_AUTHOR("Jaroslav Kysela <perex@perex.cz>");
|
2005-04-16 22:20:36 +00:00
|
|
|
MODULE_DESCRIPTION("ICEnsemble ICE1712 (Envy24)");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
|
|
|
|
static int index[SNDRV_CARDS] = SNDRV_DEFAULT_IDX; /* Index 0-MAX */
|
|
|
|
static char *id[SNDRV_CARDS] = SNDRV_DEFAULT_STR; /* ID for this card */
|
2011-12-15 03:19:36 +00:00
|
|
|
static bool enable[SNDRV_CARDS] = SNDRV_DEFAULT_ENABLE_PNP;/* Enable this card */
|
2005-04-16 22:20:36 +00:00
|
|
|
static char *model[SNDRV_CARDS];
|
2011-12-15 03:19:36 +00:00
|
|
|
static bool omni[SNDRV_CARDS]; /* Delta44 & 66 Omni I/O support */
|
2011-06-23 18:39:20 +00:00
|
|
|
static int cs8427_timeout[SNDRV_CARDS] = {[0 ... (SNDRV_CARDS-1)] = 500}; /* CS8427 S/PDIF transceiver reset timeout value in msec */
|
2006-03-21 08:57:36 +00:00
|
|
|
static int dxr_enable[SNDRV_CARDS]; /* DXR enable for DMX6FIRE */
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
module_param_array(index, int, NULL, 0444);
|
|
|
|
MODULE_PARM_DESC(index, "Index value for ICE1712 soundcard.");
|
|
|
|
module_param_array(id, charp, NULL, 0444);
|
|
|
|
MODULE_PARM_DESC(id, "ID string for ICE1712 soundcard.");
|
|
|
|
module_param_array(enable, bool, NULL, 0444);
|
|
|
|
MODULE_PARM_DESC(enable, "Enable ICE1712 soundcard.");
|
|
|
|
module_param_array(omni, bool, NULL, 0444);
|
|
|
|
MODULE_PARM_DESC(omni, "Enable Midiman M-Audio Delta Omni I/O support.");
|
|
|
|
module_param_array(cs8427_timeout, int, NULL, 0444);
|
|
|
|
MODULE_PARM_DESC(cs8427_timeout, "Define reset timeout for cs8427 chip in msec resolution.");
|
|
|
|
module_param_array(model, charp, NULL, 0444);
|
|
|
|
MODULE_PARM_DESC(model, "Use the given board model.");
|
2006-02-08 06:40:33 +00:00
|
|
|
module_param_array(dxr_enable, int, NULL, 0444);
|
2006-03-21 08:57:36 +00:00
|
|
|
MODULE_PARM_DESC(dxr_enable, "Enable DXR support for Terratec DMX6FIRE.");
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
|
2014-08-08 13:56:03 +00:00
|
|
|
static const struct pci_device_id snd_ice1712_ids[] = {
|
2009-06-25 05:13:35 +00:00
|
|
|
{ PCI_VDEVICE(ICE, PCI_DEVICE_ID_ICE_1712), 0 }, /* ICE1712 */
|
2005-04-16 22:20:36 +00:00
|
|
|
{ 0, }
|
|
|
|
};
|
|
|
|
|
|
|
|
MODULE_DEVICE_TABLE(pci, snd_ice1712_ids);
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_build_pro_mixer(struct snd_ice1712 *ice);
|
|
|
|
static int snd_ice1712_build_controls(struct snd_ice1712 *ice);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
static int PRO_RATE_LOCKED;
|
|
|
|
static int PRO_RATE_RESET = 1;
|
|
|
|
static unsigned int PRO_RATE_DEFAULT = 44100;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Basic I/O
|
|
|
|
*/
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
/* check whether the clock mode is spdif-in */
|
2005-11-17 13:59:52 +00:00
|
|
|
static inline int is_spdif_master(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
return (inb(ICEMT(ice, RATE)) & ICE1712_SPDIF_MASTER) ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static inline int is_pro_rate_locked(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
return is_spdif_master(ice) || PRO_RATE_LOCKED;
|
|
|
|
}
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
static inline void snd_ice1712_ds_write(struct snd_ice1712 *ice, u8 channel, u8 addr, u32 data)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
outb((channel << 4) | addr, ICEDS(ice, INDEX));
|
|
|
|
outl(data, ICEDS(ice, DATA));
|
|
|
|
}
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
static inline u32 snd_ice1712_ds_read(struct snd_ice1712 *ice, u8 channel, u8 addr)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
outb((channel << 4) | addr, ICEDS(ice, INDEX));
|
|
|
|
return inl(ICEDS(ice, DATA));
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static void snd_ice1712_ac97_write(struct snd_ac97 *ac97,
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned short reg,
|
|
|
|
unsigned short val)
|
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = ac97->private_data;
|
2005-04-16 22:20:36 +00:00
|
|
|
int tm;
|
|
|
|
unsigned char old_cmd = 0;
|
|
|
|
|
|
|
|
for (tm = 0; tm < 0x10000; tm++) {
|
|
|
|
old_cmd = inb(ICEREG(ice, AC97_CMD));
|
|
|
|
if (old_cmd & (ICE1712_AC97_WRITE | ICE1712_AC97_READ))
|
|
|
|
continue;
|
|
|
|
if (!(old_cmd & ICE1712_AC97_READY))
|
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
outb(reg, ICEREG(ice, AC97_INDEX));
|
|
|
|
outw(val, ICEREG(ice, AC97_DATA));
|
|
|
|
old_cmd &= ~(ICE1712_AC97_PBK_VSR | ICE1712_AC97_CAP_VSR);
|
|
|
|
outb(old_cmd | ICE1712_AC97_WRITE, ICEREG(ice, AC97_CMD));
|
|
|
|
for (tm = 0; tm < 0x10000; tm++)
|
|
|
|
if ((inb(ICEREG(ice, AC97_CMD)) & ICE1712_AC97_WRITE) == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static unsigned short snd_ice1712_ac97_read(struct snd_ac97 *ac97,
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned short reg)
|
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = ac97->private_data;
|
2005-04-16 22:20:36 +00:00
|
|
|
int tm;
|
|
|
|
unsigned char old_cmd = 0;
|
|
|
|
|
|
|
|
for (tm = 0; tm < 0x10000; tm++) {
|
|
|
|
old_cmd = inb(ICEREG(ice, AC97_CMD));
|
|
|
|
if (old_cmd & (ICE1712_AC97_WRITE | ICE1712_AC97_READ))
|
|
|
|
continue;
|
|
|
|
if (!(old_cmd & ICE1712_AC97_READY))
|
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
outb(reg, ICEREG(ice, AC97_INDEX));
|
|
|
|
outb(old_cmd | ICE1712_AC97_READ, ICEREG(ice, AC97_CMD));
|
|
|
|
for (tm = 0; tm < 0x10000; tm++)
|
|
|
|
if ((inb(ICEREG(ice, AC97_CMD)) & ICE1712_AC97_READ) == 0)
|
|
|
|
break;
|
|
|
|
if (tm >= 0x10000) /* timeout */
|
|
|
|
return ~0;
|
|
|
|
return inw(ICEREG(ice, AC97_DATA));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* pro ac97 section
|
|
|
|
*/
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static void snd_ice1712_pro_ac97_write(struct snd_ac97 *ac97,
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned short reg,
|
|
|
|
unsigned short val)
|
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = ac97->private_data;
|
2005-04-16 22:20:36 +00:00
|
|
|
int tm;
|
|
|
|
unsigned char old_cmd = 0;
|
|
|
|
|
|
|
|
for (tm = 0; tm < 0x10000; tm++) {
|
|
|
|
old_cmd = inb(ICEMT(ice, AC97_CMD));
|
|
|
|
if (old_cmd & (ICE1712_AC97_WRITE | ICE1712_AC97_READ))
|
|
|
|
continue;
|
|
|
|
if (!(old_cmd & ICE1712_AC97_READY))
|
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
outb(reg, ICEMT(ice, AC97_INDEX));
|
|
|
|
outw(val, ICEMT(ice, AC97_DATA));
|
|
|
|
old_cmd &= ~(ICE1712_AC97_PBK_VSR | ICE1712_AC97_CAP_VSR);
|
|
|
|
outb(old_cmd | ICE1712_AC97_WRITE, ICEMT(ice, AC97_CMD));
|
|
|
|
for (tm = 0; tm < 0x10000; tm++)
|
|
|
|
if ((inb(ICEMT(ice, AC97_CMD)) & ICE1712_AC97_WRITE) == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static unsigned short snd_ice1712_pro_ac97_read(struct snd_ac97 *ac97,
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned short reg)
|
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = ac97->private_data;
|
2005-04-16 22:20:36 +00:00
|
|
|
int tm;
|
|
|
|
unsigned char old_cmd = 0;
|
|
|
|
|
|
|
|
for (tm = 0; tm < 0x10000; tm++) {
|
|
|
|
old_cmd = inb(ICEMT(ice, AC97_CMD));
|
|
|
|
if (old_cmd & (ICE1712_AC97_WRITE | ICE1712_AC97_READ))
|
|
|
|
continue;
|
|
|
|
if (!(old_cmd & ICE1712_AC97_READY))
|
|
|
|
continue;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
outb(reg, ICEMT(ice, AC97_INDEX));
|
|
|
|
outb(old_cmd | ICE1712_AC97_READ, ICEMT(ice, AC97_CMD));
|
|
|
|
for (tm = 0; tm < 0x10000; tm++)
|
|
|
|
if ((inb(ICEMT(ice, AC97_CMD)) & ICE1712_AC97_READ) == 0)
|
|
|
|
break;
|
|
|
|
if (tm >= 0x10000) /* timeout */
|
|
|
|
return ~0;
|
|
|
|
return inw(ICEMT(ice, AC97_DATA));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* consumer ac97 digital mix
|
|
|
|
*/
|
2007-07-23 13:42:26 +00:00
|
|
|
#define snd_ice1712_digmix_route_ac97_info snd_ctl_boolean_mono_info
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_digmix_route_ac97_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
ucontrol->value.integer.value[0] = inb(ICEMT(ice, MONITOR_ROUTECTRL)) & ICE1712_ROUTE_AC97 ? 1 : 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_digmix_route_ac97_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned char val, nval;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
val = inb(ICEMT(ice, MONITOR_ROUTECTRL));
|
|
|
|
nval = val & ~ICE1712_ROUTE_AC97;
|
2008-09-07 10:17:02 +00:00
|
|
|
if (ucontrol->value.integer.value[0])
|
|
|
|
nval |= ICE1712_ROUTE_AC97;
|
2005-04-16 22:20:36 +00:00
|
|
|
outb(nval, ICEMT(ice, MONITOR_ROUTECTRL));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return val != nval;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_mixer_digmix_route_ac97 = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Digital Mixer To AC97",
|
|
|
|
.info = snd_ice1712_digmix_route_ac97_info,
|
|
|
|
.get = snd_ice1712_digmix_route_ac97_get,
|
|
|
|
.put = snd_ice1712_digmix_route_ac97_put,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* gpio operations
|
|
|
|
*/
|
2005-11-17 13:59:52 +00:00
|
|
|
static void snd_ice1712_set_gpio_dir(struct snd_ice1712 *ice, unsigned int data)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_DIRECTION, data);
|
|
|
|
inb(ICEREG(ice, DATA)); /* dummy read for pci-posting */
|
|
|
|
}
|
|
|
|
|
2009-09-16 20:25:38 +00:00
|
|
|
static unsigned int snd_ice1712_get_gpio_dir(struct snd_ice1712 *ice)
|
|
|
|
{
|
|
|
|
return snd_ice1712_read(ice, ICE1712_IREG_GPIO_DIRECTION);
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int snd_ice1712_get_gpio_mask(struct snd_ice1712 *ice)
|
|
|
|
{
|
|
|
|
return snd_ice1712_read(ice, ICE1712_IREG_GPIO_WRITE_MASK);
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static void snd_ice1712_set_gpio_mask(struct snd_ice1712 *ice, unsigned int data)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_WRITE_MASK, data);
|
|
|
|
inb(ICEREG(ice, DATA)); /* dummy read for pci-posting */
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static unsigned int snd_ice1712_get_gpio_data(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
return snd_ice1712_read(ice, ICE1712_IREG_GPIO_DATA);
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static void snd_ice1712_set_gpio_data(struct snd_ice1712 *ice, unsigned int val)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_DATA, val);
|
|
|
|
inb(ICEREG(ice, DATA)); /* dummy read for pci-posting */
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* CS8427 interface
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* change the input clock selection
|
|
|
|
* spdif_clock = 1 - IEC958 input, 0 - Envy24
|
|
|
|
*/
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_cs8427_set_input_clock(struct snd_ice1712 *ice, int spdif_clock)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
unsigned char reg[2] = { 0x80 | 4, 0 }; /* CS8427 auto increment | register number 4 + data */
|
|
|
|
unsigned char val, nval;
|
|
|
|
int res = 0;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
snd_i2c_lock(ice->i2c);
|
|
|
|
if (snd_i2c_sendbytes(ice->cs8427, reg, 1) != 1) {
|
|
|
|
snd_i2c_unlock(ice->i2c);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
if (snd_i2c_readbytes(ice->cs8427, &val, 1) != 1) {
|
|
|
|
snd_i2c_unlock(ice->i2c);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
nval = val & 0xf0;
|
|
|
|
if (spdif_clock)
|
|
|
|
nval |= 0x01;
|
|
|
|
else
|
|
|
|
nval |= 0x04;
|
|
|
|
if (val != nval) {
|
|
|
|
reg[1] = nval;
|
|
|
|
if (snd_i2c_sendbytes(ice->cs8427, reg, 2) != 2) {
|
|
|
|
res = -EIO;
|
|
|
|
} else {
|
|
|
|
res++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
snd_i2c_unlock(ice->i2c);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* spdif callbacks
|
|
|
|
*/
|
2005-11-17 13:59:52 +00:00
|
|
|
static void open_cs8427(struct snd_ice1712 *ice, struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
snd_cs8427_iec958_active(ice->cs8427, 1);
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static void close_cs8427(struct snd_ice1712 *ice, struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
snd_cs8427_iec958_active(ice->cs8427, 0);
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static void setup_cs8427(struct snd_ice1712 *ice, int rate)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
snd_cs8427_iec958_pcm(ice->cs8427, rate);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* create and initialize callbacks for cs8427 interface
|
|
|
|
*/
|
2012-12-06 17:35:10 +00:00
|
|
|
int snd_ice1712_init_cs8427(struct snd_ice1712 *ice, int addr)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
err = snd_cs8427_create(ice->i2c, addr,
|
|
|
|
(ice->cs8427_timeout * HZ) / 1000, &ice->cs8427);
|
|
|
|
if (err < 0) {
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_err(ice->card->dev, "CS8427 initialization failed\n");
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
ice->spdif.ops.open = open_cs8427;
|
|
|
|
ice->spdif.ops.close = close_cs8427;
|
|
|
|
ice->spdif.ops.setup_rate = setup_cs8427;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-02-02 06:56:54 +00:00
|
|
|
static void snd_ice1712_set_input_clock_source(struct snd_ice1712 *ice, int spdif_is_master)
|
|
|
|
{
|
2008-09-07 10:17:02 +00:00
|
|
|
/* change CS8427 clock source too */
|
|
|
|
if (ice->cs8427)
|
|
|
|
snd_ice1712_cs8427_set_input_clock(ice, spdif_is_master);
|
2006-02-02 06:56:54 +00:00
|
|
|
/* notify ak4524 chip as well */
|
|
|
|
if (spdif_is_master) {
|
|
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < ice->akm_codecs; i++) {
|
|
|
|
if (ice->akm[i].ops.set_rate_val)
|
|
|
|
ice->akm[i].ops.set_rate_val(&ice->akm[i], 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Interrupt handler
|
|
|
|
*/
|
|
|
|
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 13:55:46 +00:00
|
|
|
static irqreturn_t snd_ice1712_interrupt(int irq, void *dev_id)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = dev_id;
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned char status;
|
|
|
|
int handled = 0;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
status = inb(ICEREG(ice, IRQSTAT));
|
|
|
|
if (status == 0)
|
|
|
|
break;
|
|
|
|
handled = 1;
|
|
|
|
if (status & ICE1712_IRQ_MPU1) {
|
|
|
|
if (ice->rmidi[0])
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 13:55:46 +00:00
|
|
|
snd_mpu401_uart_interrupt(irq, ice->rmidi[0]->private_data);
|
2005-04-16 22:20:36 +00:00
|
|
|
outb(ICE1712_IRQ_MPU1, ICEREG(ice, IRQSTAT));
|
|
|
|
status &= ~ICE1712_IRQ_MPU1;
|
|
|
|
}
|
|
|
|
if (status & ICE1712_IRQ_TIMER)
|
|
|
|
outb(ICE1712_IRQ_TIMER, ICEREG(ice, IRQSTAT));
|
|
|
|
if (status & ICE1712_IRQ_MPU2) {
|
|
|
|
if (ice->rmidi[1])
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 13:55:46 +00:00
|
|
|
snd_mpu401_uart_interrupt(irq, ice->rmidi[1]->private_data);
|
2005-04-16 22:20:36 +00:00
|
|
|
outb(ICE1712_IRQ_MPU2, ICEREG(ice, IRQSTAT));
|
|
|
|
status &= ~ICE1712_IRQ_MPU2;
|
|
|
|
}
|
|
|
|
if (status & ICE1712_IRQ_PROPCM) {
|
|
|
|
unsigned char mtstat = inb(ICEMT(ice, IRQ));
|
|
|
|
if (mtstat & ICE1712_MULTI_PBKSTATUS) {
|
|
|
|
if (ice->playback_pro_substream)
|
|
|
|
snd_pcm_period_elapsed(ice->playback_pro_substream);
|
|
|
|
outb(ICE1712_MULTI_PBKSTATUS, ICEMT(ice, IRQ));
|
|
|
|
}
|
|
|
|
if (mtstat & ICE1712_MULTI_CAPSTATUS) {
|
|
|
|
if (ice->capture_pro_substream)
|
|
|
|
snd_pcm_period_elapsed(ice->capture_pro_substream);
|
|
|
|
outb(ICE1712_MULTI_CAPSTATUS, ICEMT(ice, IRQ));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (status & ICE1712_IRQ_FM)
|
|
|
|
outb(ICE1712_IRQ_FM, ICEREG(ice, IRQSTAT));
|
|
|
|
if (status & ICE1712_IRQ_PBKDS) {
|
|
|
|
u32 idx;
|
|
|
|
u16 pbkstatus;
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm_substream *substream;
|
2005-04-16 22:20:36 +00:00
|
|
|
pbkstatus = inw(ICEDS(ice, INTSTAT));
|
2014-02-25 16:16:16 +00:00
|
|
|
/* dev_dbg(ice->card->dev, "pbkstatus = 0x%x\n", pbkstatus); */
|
2005-04-16 22:20:36 +00:00
|
|
|
for (idx = 0; idx < 6; idx++) {
|
|
|
|
if ((pbkstatus & (3 << (idx * 2))) == 0)
|
|
|
|
continue;
|
2008-09-07 10:17:02 +00:00
|
|
|
substream = ice->playback_con_substream_ds[idx];
|
|
|
|
if (substream != NULL)
|
2005-04-16 22:20:36 +00:00
|
|
|
snd_pcm_period_elapsed(substream);
|
|
|
|
outw(3 << (idx * 2), ICEDS(ice, INTSTAT));
|
|
|
|
}
|
|
|
|
outb(ICE1712_IRQ_PBKDS, ICEREG(ice, IRQSTAT));
|
|
|
|
}
|
|
|
|
if (status & ICE1712_IRQ_CONCAP) {
|
|
|
|
if (ice->capture_con_substream)
|
|
|
|
snd_pcm_period_elapsed(ice->capture_con_substream);
|
|
|
|
outb(ICE1712_IRQ_CONCAP, ICEREG(ice, IRQSTAT));
|
|
|
|
}
|
|
|
|
if (status & ICE1712_IRQ_CONPBK) {
|
|
|
|
if (ice->playback_con_substream)
|
|
|
|
snd_pcm_period_elapsed(ice->playback_con_substream);
|
|
|
|
outb(ICE1712_IRQ_CONPBK, ICEREG(ice, IRQSTAT));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return IRQ_RETVAL(handled);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PCM part - consumer I/O
|
|
|
|
*/
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_trigger(struct snd_pcm_substream *substream,
|
2005-04-16 22:20:36 +00:00
|
|
|
int cmd)
|
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
int result = 0;
|
|
|
|
u32 tmp;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock(&ice->reg_lock);
|
|
|
|
tmp = snd_ice1712_read(ice, ICE1712_IREG_PBK_CTRL);
|
|
|
|
if (cmd == SNDRV_PCM_TRIGGER_START) {
|
|
|
|
tmp |= 1;
|
|
|
|
} else if (cmd == SNDRV_PCM_TRIGGER_STOP) {
|
|
|
|
tmp &= ~1;
|
|
|
|
} else if (cmd == SNDRV_PCM_TRIGGER_PAUSE_PUSH) {
|
|
|
|
tmp |= 2;
|
|
|
|
} else if (cmd == SNDRV_PCM_TRIGGER_PAUSE_RELEASE) {
|
|
|
|
tmp &= ~2;
|
|
|
|
} else {
|
|
|
|
result = -EINVAL;
|
|
|
|
}
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_CTRL, tmp);
|
|
|
|
spin_unlock(&ice->reg_lock);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_ds_trigger(struct snd_pcm_substream *substream,
|
2005-04-16 22:20:36 +00:00
|
|
|
int cmd)
|
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
int result = 0;
|
|
|
|
u32 tmp;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock(&ice->reg_lock);
|
|
|
|
tmp = snd_ice1712_ds_read(ice, substream->number * 2, ICE1712_DSC_CONTROL);
|
|
|
|
if (cmd == SNDRV_PCM_TRIGGER_START) {
|
|
|
|
tmp |= 1;
|
|
|
|
} else if (cmd == SNDRV_PCM_TRIGGER_STOP) {
|
|
|
|
tmp &= ~1;
|
|
|
|
} else if (cmd == SNDRV_PCM_TRIGGER_PAUSE_PUSH) {
|
|
|
|
tmp |= 2;
|
|
|
|
} else if (cmd == SNDRV_PCM_TRIGGER_PAUSE_RELEASE) {
|
|
|
|
tmp &= ~2;
|
|
|
|
} else {
|
|
|
|
result = -EINVAL;
|
|
|
|
}
|
|
|
|
snd_ice1712_ds_write(ice, substream->number * 2, ICE1712_DSC_CONTROL, tmp);
|
|
|
|
spin_unlock(&ice->reg_lock);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_capture_trigger(struct snd_pcm_substream *substream,
|
2005-04-16 22:20:36 +00:00
|
|
|
int cmd)
|
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
int result = 0;
|
|
|
|
u8 tmp;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock(&ice->reg_lock);
|
|
|
|
tmp = snd_ice1712_read(ice, ICE1712_IREG_CAP_CTRL);
|
|
|
|
if (cmd == SNDRV_PCM_TRIGGER_START) {
|
|
|
|
tmp |= 1;
|
|
|
|
} else if (cmd == SNDRV_PCM_TRIGGER_STOP) {
|
|
|
|
tmp &= ~1;
|
|
|
|
} else {
|
|
|
|
result = -EINVAL;
|
|
|
|
}
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_CAP_CTRL, tmp);
|
|
|
|
spin_unlock(&ice->reg_lock);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_prepare(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
2005-04-16 22:20:36 +00:00
|
|
|
u32 period_size, buf_size, rate, tmp;
|
|
|
|
|
|
|
|
period_size = (snd_pcm_lib_period_bytes(substream) >> 2) - 1;
|
|
|
|
buf_size = snd_pcm_lib_buffer_bytes(substream) - 1;
|
|
|
|
tmp = 0x0000;
|
|
|
|
if (snd_pcm_format_width(runtime->format) == 16)
|
|
|
|
tmp |= 0x10;
|
|
|
|
if (runtime->channels == 2)
|
|
|
|
tmp |= 0x08;
|
|
|
|
rate = (runtime->rate * 8192) / 375;
|
|
|
|
if (rate > 0x000fffff)
|
|
|
|
rate = 0x000fffff;
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
outb(0, ice->ddma_port + 15);
|
|
|
|
outb(ICE1712_DMA_MODE_WRITE | ICE1712_DMA_AUTOINIT, ice->ddma_port + 0x0b);
|
|
|
|
outl(runtime->dma_addr, ice->ddma_port + 0);
|
|
|
|
outw(buf_size, ice->ddma_port + 4);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_RATE_LO, rate & 0xff);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_RATE_MID, (rate >> 8) & 0xff);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_RATE_HI, (rate >> 16) & 0xff);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_CTRL, tmp);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_COUNT_LO, period_size & 0xff);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_COUNT_HI, period_size >> 8);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_LEFT, 0);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PBK_RIGHT, 0);
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_ds_prepare(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
2014-11-14 10:39:07 +00:00
|
|
|
u32 period_size, rate, tmp, chn;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
period_size = snd_pcm_lib_period_bytes(substream) - 1;
|
|
|
|
tmp = 0x0064;
|
|
|
|
if (snd_pcm_format_width(runtime->format) == 16)
|
|
|
|
tmp &= ~0x04;
|
|
|
|
if (runtime->channels == 2)
|
|
|
|
tmp |= 0x08;
|
|
|
|
rate = (runtime->rate * 8192) / 375;
|
|
|
|
if (rate > 0x000fffff)
|
|
|
|
rate = 0x000fffff;
|
|
|
|
ice->playback_con_active_buf[substream->number] = 0;
|
|
|
|
ice->playback_con_virt_addr[substream->number] = runtime->dma_addr;
|
|
|
|
chn = substream->number * 2;
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
snd_ice1712_ds_write(ice, chn, ICE1712_DSC_ADDR0, runtime->dma_addr);
|
|
|
|
snd_ice1712_ds_write(ice, chn, ICE1712_DSC_COUNT0, period_size);
|
|
|
|
snd_ice1712_ds_write(ice, chn, ICE1712_DSC_ADDR1, runtime->dma_addr + (runtime->periods > 1 ? period_size + 1 : 0));
|
|
|
|
snd_ice1712_ds_write(ice, chn, ICE1712_DSC_COUNT1, period_size);
|
|
|
|
snd_ice1712_ds_write(ice, chn, ICE1712_DSC_RATE, rate);
|
|
|
|
snd_ice1712_ds_write(ice, chn, ICE1712_DSC_VOLUME, 0);
|
|
|
|
snd_ice1712_ds_write(ice, chn, ICE1712_DSC_CONTROL, tmp);
|
|
|
|
if (runtime->channels == 2) {
|
|
|
|
snd_ice1712_ds_write(ice, chn + 1, ICE1712_DSC_RATE, rate);
|
|
|
|
snd_ice1712_ds_write(ice, chn + 1, ICE1712_DSC_VOLUME, 0);
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_capture_prepare(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
2005-04-16 22:20:36 +00:00
|
|
|
u32 period_size, buf_size;
|
|
|
|
u8 tmp;
|
|
|
|
|
|
|
|
period_size = (snd_pcm_lib_period_bytes(substream) >> 2) - 1;
|
|
|
|
buf_size = snd_pcm_lib_buffer_bytes(substream) - 1;
|
|
|
|
tmp = 0x06;
|
|
|
|
if (snd_pcm_format_width(runtime->format) == 16)
|
|
|
|
tmp &= ~0x04;
|
|
|
|
if (runtime->channels == 2)
|
|
|
|
tmp &= ~0x02;
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
outl(ice->capture_con_virt_addr = runtime->dma_addr, ICEREG(ice, CONCAP_ADDR));
|
|
|
|
outw(buf_size, ICEREG(ice, CONCAP_COUNT));
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_CAP_COUNT_HI, period_size >> 8);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_CAP_COUNT_LO, period_size & 0xff);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_CAP_CTRL, tmp);
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
snd_ac97_set_rate(ice->ac97, AC97_PCM_LR_ADC_RATE, runtime->rate);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static snd_pcm_uframes_t snd_ice1712_playback_pointer(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
2005-04-16 22:20:36 +00:00
|
|
|
size_t ptr;
|
|
|
|
|
|
|
|
if (!(snd_ice1712_read(ice, ICE1712_IREG_PBK_CTRL) & 1))
|
|
|
|
return 0;
|
|
|
|
ptr = runtime->buffer_size - inw(ice->ddma_port + 4);
|
2014-04-08 14:58:34 +00:00
|
|
|
ptr = bytes_to_frames(substream->runtime, ptr);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ptr == runtime->buffer_size)
|
|
|
|
ptr = 0;
|
2014-04-08 14:58:34 +00:00
|
|
|
return ptr;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static snd_pcm_uframes_t snd_ice1712_playback_ds_pointer(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
u8 addr;
|
|
|
|
size_t ptr;
|
|
|
|
|
|
|
|
if (!(snd_ice1712_ds_read(ice, substream->number * 2, ICE1712_DSC_CONTROL) & 1))
|
|
|
|
return 0;
|
|
|
|
if (ice->playback_con_active_buf[substream->number])
|
|
|
|
addr = ICE1712_DSC_ADDR1;
|
|
|
|
else
|
|
|
|
addr = ICE1712_DSC_ADDR0;
|
|
|
|
ptr = snd_ice1712_ds_read(ice, substream->number * 2, addr) -
|
|
|
|
ice->playback_con_virt_addr[substream->number];
|
2014-04-08 14:58:34 +00:00
|
|
|
ptr = bytes_to_frames(substream->runtime, ptr);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ptr == substream->runtime->buffer_size)
|
|
|
|
ptr = 0;
|
2014-04-08 14:58:34 +00:00
|
|
|
return ptr;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static snd_pcm_uframes_t snd_ice1712_capture_pointer(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
size_t ptr;
|
|
|
|
|
|
|
|
if (!(snd_ice1712_read(ice, ICE1712_IREG_CAP_CTRL) & 1))
|
|
|
|
return 0;
|
|
|
|
ptr = inl(ICEREG(ice, CONCAP_ADDR)) - ice->capture_con_virt_addr;
|
2014-04-08 14:58:34 +00:00
|
|
|
ptr = bytes_to_frames(substream->runtime, ptr);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ptr == substream->runtime->buffer_size)
|
|
|
|
ptr = 0;
|
2014-04-08 14:58:34 +00:00
|
|
|
return ptr;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
static const struct snd_pcm_hardware snd_ice1712_playback = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = (SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_INTERLEAVED |
|
|
|
|
SNDRV_PCM_INFO_BLOCK_TRANSFER |
|
|
|
|
SNDRV_PCM_INFO_MMAP_VALID |
|
|
|
|
SNDRV_PCM_INFO_PAUSE),
|
|
|
|
.formats = SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_S16_LE,
|
|
|
|
.rates = SNDRV_PCM_RATE_CONTINUOUS | SNDRV_PCM_RATE_8000_48000,
|
|
|
|
.rate_min = 4000,
|
|
|
|
.rate_max = 48000,
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.buffer_bytes_max = (64*1024),
|
|
|
|
.period_bytes_min = 64,
|
|
|
|
.period_bytes_max = (64*1024),
|
|
|
|
.periods_min = 1,
|
|
|
|
.periods_max = 1024,
|
|
|
|
.fifo_size = 0,
|
|
|
|
};
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
static const struct snd_pcm_hardware snd_ice1712_playback_ds = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = (SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_INTERLEAVED |
|
|
|
|
SNDRV_PCM_INFO_BLOCK_TRANSFER |
|
|
|
|
SNDRV_PCM_INFO_MMAP_VALID |
|
|
|
|
SNDRV_PCM_INFO_PAUSE),
|
|
|
|
.formats = SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_S16_LE,
|
|
|
|
.rates = SNDRV_PCM_RATE_CONTINUOUS | SNDRV_PCM_RATE_8000_48000,
|
|
|
|
.rate_min = 4000,
|
|
|
|
.rate_max = 48000,
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.buffer_bytes_max = (128*1024),
|
|
|
|
.period_bytes_min = 64,
|
|
|
|
.period_bytes_max = (128*1024),
|
|
|
|
.periods_min = 2,
|
|
|
|
.periods_max = 2,
|
|
|
|
.fifo_size = 0,
|
|
|
|
};
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
static const struct snd_pcm_hardware snd_ice1712_capture = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = (SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_INTERLEAVED |
|
|
|
|
SNDRV_PCM_INFO_BLOCK_TRANSFER |
|
|
|
|
SNDRV_PCM_INFO_MMAP_VALID),
|
|
|
|
.formats = SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_S16_LE,
|
|
|
|
.rates = SNDRV_PCM_RATE_CONTINUOUS | SNDRV_PCM_RATE_8000_48000,
|
|
|
|
.rate_min = 4000,
|
|
|
|
.rate_max = 48000,
|
|
|
|
.channels_min = 1,
|
|
|
|
.channels_max = 2,
|
|
|
|
.buffer_bytes_max = (64*1024),
|
|
|
|
.period_bytes_min = 64,
|
|
|
|
.period_bytes_max = (64*1024),
|
|
|
|
.periods_min = 1,
|
|
|
|
.periods_max = 1024,
|
|
|
|
.fifo_size = 0,
|
|
|
|
};
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_open(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->playback_con_substream = substream;
|
|
|
|
runtime->hw = snd_ice1712_playback;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_ds_open(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
u32 tmp;
|
|
|
|
|
|
|
|
ice->playback_con_substream_ds[substream->number] = substream;
|
|
|
|
runtime->hw = snd_ice1712_playback_ds;
|
2008-09-07 10:17:02 +00:00
|
|
|
spin_lock_irq(&ice->reg_lock);
|
2005-04-16 22:20:36 +00:00
|
|
|
tmp = inw(ICEDS(ice, INTMASK)) & ~(1 << (substream->number * 2));
|
|
|
|
outw(tmp, ICEDS(ice, INTMASK));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_capture_open(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->capture_con_substream = substream;
|
|
|
|
runtime->hw = snd_ice1712_capture;
|
|
|
|
runtime->hw.rates = ice->ac97->rates[AC97_RATES_ADC];
|
|
|
|
if (!(runtime->hw.rates & SNDRV_PCM_RATE_8000))
|
|
|
|
runtime->hw.rate_min = 48000;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_close(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->playback_con_substream = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_ds_close(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
u32 tmp;
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
spin_lock_irq(&ice->reg_lock);
|
2005-04-16 22:20:36 +00:00
|
|
|
tmp = inw(ICEDS(ice, INTMASK)) | (3 << (substream->number * 2));
|
|
|
|
outw(tmp, ICEDS(ice, INTMASK));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
ice->playback_con_substream_ds[substream->number] = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_capture_close(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->capture_con_substream = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-09-01 22:13:10 +00:00
|
|
|
static const struct snd_pcm_ops snd_ice1712_playback_ops = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.open = snd_ice1712_playback_open,
|
|
|
|
.close = snd_ice1712_playback_close,
|
|
|
|
.prepare = snd_ice1712_playback_prepare,
|
|
|
|
.trigger = snd_ice1712_playback_trigger,
|
|
|
|
.pointer = snd_ice1712_playback_pointer,
|
|
|
|
};
|
|
|
|
|
2016-09-01 22:13:10 +00:00
|
|
|
static const struct snd_pcm_ops snd_ice1712_playback_ds_ops = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.open = snd_ice1712_playback_ds_open,
|
|
|
|
.close = snd_ice1712_playback_ds_close,
|
|
|
|
.prepare = snd_ice1712_playback_ds_prepare,
|
|
|
|
.trigger = snd_ice1712_playback_ds_trigger,
|
|
|
|
.pointer = snd_ice1712_playback_ds_pointer,
|
|
|
|
};
|
|
|
|
|
2016-09-01 22:13:10 +00:00
|
|
|
static const struct snd_pcm_ops snd_ice1712_capture_ops = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.open = snd_ice1712_capture_open,
|
|
|
|
.close = snd_ice1712_capture_close,
|
|
|
|
.prepare = snd_ice1712_capture_prepare,
|
|
|
|
.trigger = snd_ice1712_capture_trigger,
|
|
|
|
.pointer = snd_ice1712_capture_pointer,
|
|
|
|
};
|
|
|
|
|
2015-01-02 11:24:51 +00:00
|
|
|
static int snd_ice1712_pcm(struct snd_ice1712 *ice, int device)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm *pcm;
|
2005-04-16 22:20:36 +00:00
|
|
|
int err;
|
|
|
|
|
|
|
|
err = snd_pcm_new(ice->card, "ICE1712 consumer", device, 1, 1, &pcm);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_ice1712_playback_ops);
|
|
|
|
snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &snd_ice1712_capture_ops);
|
|
|
|
|
|
|
|
pcm->private_data = ice;
|
|
|
|
pcm->info_flags = 0;
|
|
|
|
strcpy(pcm->name, "ICE1712 consumer");
|
|
|
|
ice->pcm = pcm;
|
|
|
|
|
2019-12-09 09:49:13 +00:00
|
|
|
snd_pcm_set_managed_buffer_all(pcm, SNDRV_DMA_TYPE_DEV,
|
|
|
|
&ice->pci->dev, 64*1024, 64*1024);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_warn(ice->card->dev,
|
|
|
|
"Consumer PCM code does not work well at the moment --jk\n");
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-01-02 11:24:51 +00:00
|
|
|
static int snd_ice1712_pcm_ds(struct snd_ice1712 *ice, int device)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm *pcm;
|
2005-04-16 22:20:36 +00:00
|
|
|
int err;
|
|
|
|
|
|
|
|
err = snd_pcm_new(ice->card, "ICE1712 consumer (DS)", device, 6, 0, &pcm);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_ice1712_playback_ds_ops);
|
|
|
|
|
|
|
|
pcm->private_data = ice;
|
|
|
|
pcm->info_flags = 0;
|
|
|
|
strcpy(pcm->name, "ICE1712 consumer (DS)");
|
|
|
|
ice->pcm_ds = pcm;
|
|
|
|
|
2019-12-09 09:49:13 +00:00
|
|
|
snd_pcm_set_managed_buffer_all(pcm, SNDRV_DMA_TYPE_DEV,
|
|
|
|
&ice->pci->dev, 64*1024, 128*1024);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PCM code - professional part (multitrack)
|
|
|
|
*/
|
|
|
|
|
2017-06-07 12:20:40 +00:00
|
|
|
static const unsigned int rates[] = { 8000, 9600, 11025, 12000, 16000, 22050, 24000,
|
2005-04-16 22:20:36 +00:00
|
|
|
32000, 44100, 48000, 64000, 88200, 96000 };
|
|
|
|
|
2017-06-07 12:20:40 +00:00
|
|
|
static const struct snd_pcm_hw_constraint_list hw_constraints_rates = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.count = ARRAY_SIZE(rates),
|
|
|
|
.list = rates,
|
|
|
|
.mask = 0,
|
|
|
|
};
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_trigger(struct snd_pcm_substream *substream,
|
2005-04-16 22:20:36 +00:00
|
|
|
int cmd)
|
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
switch (cmd) {
|
|
|
|
case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
|
|
|
|
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
|
|
|
|
{
|
|
|
|
unsigned int what;
|
|
|
|
unsigned int old;
|
|
|
|
if (substream->stream != SNDRV_PCM_STREAM_PLAYBACK)
|
|
|
|
return -EINVAL;
|
|
|
|
what = ICE1712_PLAYBACK_PAUSE;
|
|
|
|
snd_pcm_trigger_done(substream, substream);
|
|
|
|
spin_lock(&ice->reg_lock);
|
|
|
|
old = inl(ICEMT(ice, PLAYBACK_CONTROL));
|
|
|
|
if (cmd == SNDRV_PCM_TRIGGER_PAUSE_PUSH)
|
|
|
|
old |= what;
|
|
|
|
else
|
|
|
|
old &= ~what;
|
|
|
|
outl(old, ICEMT(ice, PLAYBACK_CONTROL));
|
|
|
|
spin_unlock(&ice->reg_lock);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SNDRV_PCM_TRIGGER_START:
|
|
|
|
case SNDRV_PCM_TRIGGER_STOP:
|
|
|
|
{
|
|
|
|
unsigned int what = 0;
|
|
|
|
unsigned int old;
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm_substream *s;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2007-02-22 11:52:53 +00:00
|
|
|
snd_pcm_group_for_each_entry(s, substream) {
|
2005-04-16 22:20:36 +00:00
|
|
|
if (s == ice->playback_pro_substream) {
|
|
|
|
what |= ICE1712_PLAYBACK_START;
|
|
|
|
snd_pcm_trigger_done(s, substream);
|
|
|
|
} else if (s == ice->capture_pro_substream) {
|
|
|
|
what |= ICE1712_CAPTURE_START_SHADOW;
|
|
|
|
snd_pcm_trigger_done(s, substream);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_lock(&ice->reg_lock);
|
|
|
|
old = inl(ICEMT(ice, PLAYBACK_CONTROL));
|
|
|
|
if (cmd == SNDRV_PCM_TRIGGER_START)
|
|
|
|
old |= what;
|
|
|
|
else
|
|
|
|
old &= ~what;
|
|
|
|
outl(old, ICEMT(ice, PLAYBACK_CONTROL));
|
|
|
|
spin_unlock(&ice->reg_lock);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*/
|
2005-11-17 13:59:52 +00:00
|
|
|
static void snd_ice1712_set_pro_rate(struct snd_ice1712 *ice, unsigned int rate, int force)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
unsigned char val, old;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
switch (rate) {
|
|
|
|
case 8000: val = 6; break;
|
|
|
|
case 9600: val = 3; break;
|
|
|
|
case 11025: val = 10; break;
|
|
|
|
case 12000: val = 2; break;
|
|
|
|
case 16000: val = 5; break;
|
|
|
|
case 22050: val = 9; break;
|
|
|
|
case 24000: val = 1; break;
|
|
|
|
case 32000: val = 4; break;
|
|
|
|
case 44100: val = 8; break;
|
|
|
|
case 48000: val = 0; break;
|
|
|
|
case 64000: val = 15; break;
|
|
|
|
case 88200: val = 11; break;
|
|
|
|
case 96000: val = 7; break;
|
|
|
|
default:
|
|
|
|
snd_BUG();
|
|
|
|
val = 0;
|
|
|
|
rate = 48000;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock_irqsave(&ice->reg_lock, flags);
|
|
|
|
if (inb(ICEMT(ice, PLAYBACK_CONTROL)) & (ICE1712_CAPTURE_START_SHADOW|
|
|
|
|
ICE1712_PLAYBACK_PAUSE|
|
|
|
|
ICE1712_PLAYBACK_START)) {
|
2008-09-07 10:17:02 +00:00
|
|
|
__out:
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_unlock_irqrestore(&ice->reg_lock, flags);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!force && is_pro_rate_locked(ice))
|
|
|
|
goto __out;
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
old = inb(ICEMT(ice, RATE));
|
2005-04-16 22:20:36 +00:00
|
|
|
if (!force && old == val)
|
|
|
|
goto __out;
|
2014-04-03 21:09:38 +00:00
|
|
|
|
|
|
|
ice->cur_rate = rate;
|
2005-04-16 22:20:36 +00:00
|
|
|
outb(val, ICEMT(ice, RATE));
|
|
|
|
spin_unlock_irqrestore(&ice->reg_lock, flags);
|
|
|
|
|
|
|
|
if (ice->gpio.set_pro_rate)
|
|
|
|
ice->gpio.set_pro_rate(ice, rate);
|
|
|
|
for (i = 0; i < ice->akm_codecs; i++) {
|
|
|
|
if (ice->akm[i].ops.set_rate_val)
|
|
|
|
ice->akm[i].ops.set_rate_val(&ice->akm[i], rate);
|
|
|
|
}
|
|
|
|
if (ice->spdif.ops.setup_rate)
|
|
|
|
ice->spdif.ops.setup_rate(ice, rate);
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_pro_prepare(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->playback_pro_size = snd_pcm_lib_buffer_bytes(substream);
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
outl(substream->runtime->dma_addr, ICEMT(ice, PLAYBACK_ADDR));
|
|
|
|
outw((ice->playback_pro_size >> 2) - 1, ICEMT(ice, PLAYBACK_SIZE));
|
|
|
|
outw((snd_pcm_lib_period_bytes(substream) >> 2) - 1, ICEMT(ice, PLAYBACK_COUNT));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_pro_hw_params(struct snd_pcm_substream *substream,
|
|
|
|
struct snd_pcm_hw_params *hw_params)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
snd_ice1712_set_pro_rate(ice, params_rate(hw_params), 0);
|
2019-12-09 09:49:13 +00:00
|
|
|
return 0;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_capture_pro_prepare(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->capture_pro_size = snd_pcm_lib_buffer_bytes(substream);
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
outl(substream->runtime->dma_addr, ICEMT(ice, CAPTURE_ADDR));
|
|
|
|
outw((ice->capture_pro_size >> 2) - 1, ICEMT(ice, CAPTURE_SIZE));
|
|
|
|
outw((snd_pcm_lib_period_bytes(substream) >> 2) - 1, ICEMT(ice, CAPTURE_COUNT));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_capture_pro_hw_params(struct snd_pcm_substream *substream,
|
|
|
|
struct snd_pcm_hw_params *hw_params)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
snd_ice1712_set_pro_rate(ice, params_rate(hw_params), 0);
|
2019-12-09 09:49:13 +00:00
|
|
|
return 0;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static snd_pcm_uframes_t snd_ice1712_playback_pro_pointer(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
size_t ptr;
|
|
|
|
|
|
|
|
if (!(inl(ICEMT(ice, PLAYBACK_CONTROL)) & ICE1712_PLAYBACK_START))
|
|
|
|
return 0;
|
|
|
|
ptr = ice->playback_pro_size - (inw(ICEMT(ice, PLAYBACK_SIZE)) << 2);
|
2014-04-08 14:58:34 +00:00
|
|
|
ptr = bytes_to_frames(substream->runtime, ptr);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ptr == substream->runtime->buffer_size)
|
|
|
|
ptr = 0;
|
2014-04-08 14:58:34 +00:00
|
|
|
return ptr;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static snd_pcm_uframes_t snd_ice1712_capture_pro_pointer(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
size_t ptr;
|
|
|
|
|
|
|
|
if (!(inl(ICEMT(ice, PLAYBACK_CONTROL)) & ICE1712_CAPTURE_START_SHADOW))
|
|
|
|
return 0;
|
|
|
|
ptr = ice->capture_pro_size - (inw(ICEMT(ice, CAPTURE_SIZE)) << 2);
|
2014-04-08 14:58:34 +00:00
|
|
|
ptr = bytes_to_frames(substream->runtime, ptr);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ptr == substream->runtime->buffer_size)
|
|
|
|
ptr = 0;
|
2014-04-08 14:58:34 +00:00
|
|
|
return ptr;
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
static const struct snd_pcm_hardware snd_ice1712_playback_pro = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = (SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_INTERLEAVED |
|
|
|
|
SNDRV_PCM_INFO_BLOCK_TRANSFER |
|
|
|
|
SNDRV_PCM_INFO_MMAP_VALID |
|
|
|
|
SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_SYNC_START),
|
|
|
|
.formats = SNDRV_PCM_FMTBIT_S32_LE,
|
|
|
|
.rates = SNDRV_PCM_RATE_KNOT | SNDRV_PCM_RATE_8000_96000,
|
|
|
|
.rate_min = 4000,
|
|
|
|
.rate_max = 96000,
|
|
|
|
.channels_min = 10,
|
|
|
|
.channels_max = 10,
|
|
|
|
.buffer_bytes_max = (256*1024),
|
|
|
|
.period_bytes_min = 10 * 4 * 2,
|
|
|
|
.period_bytes_max = 131040,
|
|
|
|
.periods_min = 1,
|
|
|
|
.periods_max = 1024,
|
|
|
|
.fifo_size = 0,
|
|
|
|
};
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
static const struct snd_pcm_hardware snd_ice1712_capture_pro = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = (SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_INTERLEAVED |
|
|
|
|
SNDRV_PCM_INFO_BLOCK_TRANSFER |
|
|
|
|
SNDRV_PCM_INFO_MMAP_VALID |
|
|
|
|
SNDRV_PCM_INFO_PAUSE | SNDRV_PCM_INFO_SYNC_START),
|
|
|
|
.formats = SNDRV_PCM_FMTBIT_S32_LE,
|
|
|
|
.rates = SNDRV_PCM_RATE_KNOT | SNDRV_PCM_RATE_8000_96000,
|
|
|
|
.rate_min = 4000,
|
|
|
|
.rate_max = 96000,
|
|
|
|
.channels_min = 12,
|
|
|
|
.channels_max = 12,
|
|
|
|
.buffer_bytes_max = (256*1024),
|
|
|
|
.period_bytes_min = 12 * 4 * 2,
|
|
|
|
.period_bytes_max = 131040,
|
|
|
|
.periods_min = 1,
|
|
|
|
.periods_max = 1024,
|
|
|
|
.fifo_size = 0,
|
|
|
|
};
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_pro_open(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->playback_pro_substream = substream;
|
|
|
|
runtime->hw = snd_ice1712_playback_pro;
|
|
|
|
snd_pcm_set_sync(substream);
|
|
|
|
snd_pcm_hw_constraint_msbits(runtime, 0, 32, 24);
|
|
|
|
snd_pcm_hw_constraint_list(runtime, 0, SNDRV_PCM_HW_PARAM_RATE, &hw_constraints_rates);
|
2010-02-05 07:58:20 +00:00
|
|
|
if (is_pro_rate_locked(ice)) {
|
|
|
|
runtime->hw.rate_min = PRO_RATE_DEFAULT;
|
|
|
|
runtime->hw.rate_max = PRO_RATE_DEFAULT;
|
|
|
|
}
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
if (ice->spdif.ops.open)
|
|
|
|
ice->spdif.ops.open(ice, substream);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_capture_pro_open(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
|
|
|
struct snd_pcm_runtime *runtime = substream->runtime;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->capture_pro_substream = substream;
|
|
|
|
runtime->hw = snd_ice1712_capture_pro;
|
|
|
|
snd_pcm_set_sync(substream);
|
|
|
|
snd_pcm_hw_constraint_msbits(runtime, 0, 32, 24);
|
|
|
|
snd_pcm_hw_constraint_list(runtime, 0, SNDRV_PCM_HW_PARAM_RATE, &hw_constraints_rates);
|
2010-02-05 07:58:20 +00:00
|
|
|
if (is_pro_rate_locked(ice)) {
|
|
|
|
runtime->hw.rate_min = PRO_RATE_DEFAULT;
|
|
|
|
runtime->hw.rate_max = PRO_RATE_DEFAULT;
|
|
|
|
}
|
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_playback_pro_close(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
if (PRO_RATE_RESET)
|
|
|
|
snd_ice1712_set_pro_rate(ice, PRO_RATE_DEFAULT, 0);
|
|
|
|
ice->playback_pro_substream = NULL;
|
|
|
|
if (ice->spdif.ops.close)
|
|
|
|
ice->spdif.ops.close(ice, substream);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_capture_pro_close(struct snd_pcm_substream *substream)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_pcm_substream_chip(substream);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
if (PRO_RATE_RESET)
|
|
|
|
snd_ice1712_set_pro_rate(ice, PRO_RATE_DEFAULT, 0);
|
|
|
|
ice->capture_pro_substream = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-09-01 22:13:10 +00:00
|
|
|
static const struct snd_pcm_ops snd_ice1712_playback_pro_ops = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.open = snd_ice1712_playback_pro_open,
|
|
|
|
.close = snd_ice1712_playback_pro_close,
|
|
|
|
.hw_params = snd_ice1712_playback_pro_hw_params,
|
|
|
|
.prepare = snd_ice1712_playback_pro_prepare,
|
|
|
|
.trigger = snd_ice1712_pro_trigger,
|
|
|
|
.pointer = snd_ice1712_playback_pro_pointer,
|
|
|
|
};
|
|
|
|
|
2016-09-01 22:13:10 +00:00
|
|
|
static const struct snd_pcm_ops snd_ice1712_capture_pro_ops = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.open = snd_ice1712_capture_pro_open,
|
|
|
|
.close = snd_ice1712_capture_pro_close,
|
|
|
|
.hw_params = snd_ice1712_capture_pro_hw_params,
|
|
|
|
.prepare = snd_ice1712_capture_pro_prepare,
|
|
|
|
.trigger = snd_ice1712_pro_trigger,
|
|
|
|
.pointer = snd_ice1712_capture_pro_pointer,
|
|
|
|
};
|
|
|
|
|
2015-01-02 11:24:51 +00:00
|
|
|
static int snd_ice1712_pcm_profi(struct snd_ice1712 *ice, int device)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_pcm *pcm;
|
2005-04-16 22:20:36 +00:00
|
|
|
int err;
|
|
|
|
|
|
|
|
err = snd_pcm_new(ice->card, "ICE1712 multi", device, 1, 1, &pcm);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_ice1712_playback_pro_ops);
|
|
|
|
snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &snd_ice1712_capture_pro_ops);
|
|
|
|
|
|
|
|
pcm->private_data = ice;
|
|
|
|
pcm->info_flags = 0;
|
|
|
|
strcpy(pcm->name, "ICE1712 multi");
|
|
|
|
|
2019-12-09 09:49:13 +00:00
|
|
|
snd_pcm_set_managed_buffer_all(pcm, SNDRV_DMA_TYPE_DEV,
|
|
|
|
&ice->pci->dev, 256*1024, 256*1024);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
ice->pcm_pro = pcm;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ice->cs8427) {
|
|
|
|
/* assign channels to iec958 */
|
|
|
|
err = snd_cs8427_iec958_build(ice->cs8427,
|
|
|
|
pcm->streams[0].substream,
|
|
|
|
pcm->streams[1].substream);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2014-11-14 10:39:05 +00:00
|
|
|
return snd_ice1712_build_pro_mixer(ice);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Mixer section
|
|
|
|
*/
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static void snd_ice1712_update_volume(struct snd_ice1712 *ice, int index)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
unsigned int vol = ice->pro_volumes[index];
|
|
|
|
unsigned short val = 0;
|
|
|
|
|
|
|
|
val |= (vol & 0x8000) == 0 ? (96 - (vol & 0x7f)) : 0x7f;
|
|
|
|
val |= ((vol & 0x80000000) == 0 ? (96 - ((vol >> 16) & 0x7f)) : 0x7f) << 8;
|
|
|
|
outb(index, ICEMT(ice, MONITOR_INDEX));
|
|
|
|
outw(val, ICEMT(ice, MONITOR_VOLUME));
|
|
|
|
}
|
|
|
|
|
2007-07-23 13:42:26 +00:00
|
|
|
#define snd_ice1712_pro_mixer_switch_info snd_ctl_boolean_stereo_info
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_mixer_switch_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2008-02-29 10:52:50 +00:00
|
|
|
int priv_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id) +
|
|
|
|
kcontrol->private_value;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock_irq(&ice->reg_lock);
|
2008-02-29 10:52:50 +00:00
|
|
|
ucontrol->value.integer.value[0] =
|
|
|
|
!((ice->pro_volumes[priv_idx] >> 15) & 1);
|
|
|
|
ucontrol->value.integer.value[1] =
|
|
|
|
!((ice->pro_volumes[priv_idx] >> 31) & 1);
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_mixer_switch_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2008-02-29 10:52:50 +00:00
|
|
|
int priv_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id) +
|
|
|
|
kcontrol->private_value;
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned int nval, change;
|
|
|
|
|
|
|
|
nval = (ucontrol->value.integer.value[0] ? 0 : 0x00008000) |
|
|
|
|
(ucontrol->value.integer.value[1] ? 0 : 0x80000000);
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
2008-02-29 10:52:50 +00:00
|
|
|
nval |= ice->pro_volumes[priv_idx] & ~0x80008000;
|
|
|
|
change = nval != ice->pro_volumes[priv_idx];
|
|
|
|
ice->pro_volumes[priv_idx] = nval;
|
|
|
|
snd_ice1712_update_volume(ice, priv_idx);
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_mixer_volume_info(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_info *uinfo)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
|
|
|
|
uinfo->count = 2;
|
|
|
|
uinfo->value.integer.min = 0;
|
|
|
|
uinfo->value.integer.max = 96;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_mixer_volume_get(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2008-02-29 10:52:50 +00:00
|
|
|
int priv_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id) +
|
|
|
|
kcontrol->private_value;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock_irq(&ice->reg_lock);
|
2008-02-29 10:52:50 +00:00
|
|
|
ucontrol->value.integer.value[0] =
|
|
|
|
(ice->pro_volumes[priv_idx] >> 0) & 127;
|
|
|
|
ucontrol->value.integer.value[1] =
|
|
|
|
(ice->pro_volumes[priv_idx] >> 16) & 127;
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_mixer_volume_put(struct snd_kcontrol *kcontrol, struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2008-02-29 10:52:50 +00:00
|
|
|
int priv_idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id) +
|
|
|
|
kcontrol->private_value;
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned int nval, change;
|
|
|
|
|
|
|
|
nval = (ucontrol->value.integer.value[0] & 127) |
|
|
|
|
((ucontrol->value.integer.value[1] & 127) << 16);
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
2008-02-29 10:52:50 +00:00
|
|
|
nval |= ice->pro_volumes[priv_idx] & ~0x007f007f;
|
|
|
|
change = nval != ice->pro_volumes[priv_idx];
|
|
|
|
ice->pro_volumes[priv_idx] = nval;
|
|
|
|
snd_ice1712_update_volume(ice, priv_idx);
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2007-01-29 14:33:49 +00:00
|
|
|
static const DECLARE_TLV_DB_SCALE(db_scale_playback, -14400, 150, 0);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2020-01-03 08:16:53 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_multi_playback_ctrls[] = {
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Multi Playback Switch",
|
|
|
|
.info = snd_ice1712_pro_mixer_switch_info,
|
|
|
|
.get = snd_ice1712_pro_mixer_switch_get,
|
|
|
|
.put = snd_ice1712_pro_mixer_switch_put,
|
|
|
|
.private_value = 0,
|
|
|
|
.count = 10,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
2006-08-30 14:56:30 +00:00
|
|
|
.access = (SNDRV_CTL_ELEM_ACCESS_READWRITE |
|
|
|
|
SNDRV_CTL_ELEM_ACCESS_TLV_READ),
|
2005-04-16 22:20:36 +00:00
|
|
|
.name = "Multi Playback Volume",
|
|
|
|
.info = snd_ice1712_pro_mixer_volume_info,
|
|
|
|
.get = snd_ice1712_pro_mixer_volume_get,
|
|
|
|
.put = snd_ice1712_pro_mixer_volume_put,
|
|
|
|
.private_value = 0,
|
|
|
|
.count = 10,
|
2006-08-30 14:56:30 +00:00
|
|
|
.tlv = { .p = db_scale_playback }
|
2005-04-16 22:20:36 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2017-05-17 11:50:46 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_multi_capture_analog_switch = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "H/W Multi Capture Switch",
|
|
|
|
.info = snd_ice1712_pro_mixer_switch_info,
|
|
|
|
.get = snd_ice1712_pro_mixer_switch_get,
|
|
|
|
.put = snd_ice1712_pro_mixer_switch_put,
|
|
|
|
.private_value = 10,
|
|
|
|
};
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_multi_capture_spdif_switch = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
2008-09-07 10:17:02 +00:00
|
|
|
.name = SNDRV_CTL_NAME_IEC958("Multi ", CAPTURE, SWITCH),
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = snd_ice1712_pro_mixer_switch_info,
|
|
|
|
.get = snd_ice1712_pro_mixer_switch_get,
|
|
|
|
.put = snd_ice1712_pro_mixer_switch_put,
|
|
|
|
.private_value = 18,
|
|
|
|
.count = 2,
|
|
|
|
};
|
|
|
|
|
2017-05-17 11:50:46 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_multi_capture_analog_volume = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
2006-08-30 14:56:30 +00:00
|
|
|
.access = (SNDRV_CTL_ELEM_ACCESS_READWRITE |
|
|
|
|
SNDRV_CTL_ELEM_ACCESS_TLV_READ),
|
2005-04-16 22:20:36 +00:00
|
|
|
.name = "H/W Multi Capture Volume",
|
|
|
|
.info = snd_ice1712_pro_mixer_volume_info,
|
|
|
|
.get = snd_ice1712_pro_mixer_volume_get,
|
|
|
|
.put = snd_ice1712_pro_mixer_volume_put,
|
|
|
|
.private_value = 10,
|
2006-08-30 14:56:30 +00:00
|
|
|
.tlv = { .p = db_scale_playback }
|
2005-04-16 22:20:36 +00:00
|
|
|
};
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_multi_capture_spdif_volume = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
2008-09-07 10:17:02 +00:00
|
|
|
.name = SNDRV_CTL_NAME_IEC958("Multi ", CAPTURE, VOLUME),
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = snd_ice1712_pro_mixer_volume_info,
|
|
|
|
.get = snd_ice1712_pro_mixer_volume_get,
|
|
|
|
.put = snd_ice1712_pro_mixer_volume_put,
|
|
|
|
.private_value = 18,
|
|
|
|
.count = 2,
|
|
|
|
};
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static int snd_ice1712_build_pro_mixer(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_card *card = ice->card;
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned int idx;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
/* multi-channel mixer */
|
|
|
|
for (idx = 0; idx < ARRAY_SIZE(snd_ice1712_multi_playback_ctrls); idx++) {
|
|
|
|
err = snd_ctl_add(card, snd_ctl_new1(&snd_ice1712_multi_playback_ctrls[idx], ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ice->num_total_adcs > 0) {
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_kcontrol_new tmp = snd_ice1712_multi_capture_analog_switch;
|
2005-04-16 22:20:36 +00:00
|
|
|
tmp.count = ice->num_total_adcs;
|
|
|
|
err = snd_ctl_add(card, snd_ctl_new1(&tmp, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_ctl_add(card, snd_ctl_new1(&snd_ice1712_multi_capture_spdif_switch, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (ice->num_total_adcs > 0) {
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_kcontrol_new tmp = snd_ice1712_multi_capture_analog_volume;
|
2005-04-16 22:20:36 +00:00
|
|
|
tmp.count = ice->num_total_adcs;
|
|
|
|
err = snd_ctl_add(card, snd_ctl_new1(&tmp, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_ctl_add(card, snd_ctl_new1(&snd_ice1712_multi_capture_spdif_volume, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
/* initialize volumes */
|
|
|
|
for (idx = 0; idx < 10; idx++) {
|
|
|
|
ice->pro_volumes[idx] = 0x80008000; /* mute */
|
|
|
|
snd_ice1712_update_volume(ice, idx);
|
|
|
|
}
|
|
|
|
for (idx = 10; idx < 10 + ice->num_total_adcs; idx++) {
|
|
|
|
ice->pro_volumes[idx] = 0x80008000; /* mute */
|
|
|
|
snd_ice1712_update_volume(ice, idx);
|
|
|
|
}
|
|
|
|
for (idx = 18; idx < 20; idx++) {
|
|
|
|
ice->pro_volumes[idx] = 0x80008000; /* mute */
|
|
|
|
snd_ice1712_update_volume(ice, idx);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static void snd_ice1712_mixer_free_ac97(struct snd_ac97 *ac97)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = ac97->private_data;
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->ac97 = NULL;
|
|
|
|
}
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static int snd_ice1712_ac97_mixer(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
int err, bus_num = 0;
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ac97_template ac97;
|
|
|
|
struct snd_ac97_bus *pbus;
|
2020-01-03 08:16:43 +00:00
|
|
|
static const struct snd_ac97_bus_ops con_ops = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.write = snd_ice1712_ac97_write,
|
|
|
|
.read = snd_ice1712_ac97_read,
|
|
|
|
};
|
2020-01-03 08:16:43 +00:00
|
|
|
static const struct snd_ac97_bus_ops pro_ops = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.write = snd_ice1712_pro_ac97_write,
|
|
|
|
.read = snd_ice1712_pro_ac97_read,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (ice_has_con_ac97(ice)) {
|
2008-09-07 10:17:02 +00:00
|
|
|
err = snd_ac97_bus(ice->card, bus_num++, &con_ops, NULL, &pbus);
|
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
memset(&ac97, 0, sizeof(ac97));
|
|
|
|
ac97.private_data = ice;
|
|
|
|
ac97.private_free = snd_ice1712_mixer_free_ac97;
|
2008-09-07 10:17:02 +00:00
|
|
|
err = snd_ac97_mixer(pbus, &ac97, &ice->ac97);
|
|
|
|
if (err < 0)
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_warn(ice->card->dev,
|
|
|
|
"cannot initialize ac97 for consumer, skipped\n");
|
2005-04-16 22:20:36 +00:00
|
|
|
else {
|
2014-11-14 10:39:05 +00:00
|
|
|
return snd_ctl_add(ice->card,
|
|
|
|
snd_ctl_new1(&snd_ice1712_mixer_digmix_route_ac97,
|
|
|
|
ice));
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
if (!(ice->eeprom.data[ICE_EEP1_ACLINK] & ICE1712_CFG_PRO_I2S)) {
|
|
|
|
err = snd_ac97_bus(ice->card, bus_num, &pro_ops, NULL, &pbus);
|
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
memset(&ac97, 0, sizeof(ac97));
|
|
|
|
ac97.private_data = ice;
|
|
|
|
ac97.private_free = snd_ice1712_mixer_free_ac97;
|
2008-09-07 10:17:02 +00:00
|
|
|
err = snd_ac97_mixer(pbus, &ac97, &ice->ac97);
|
|
|
|
if (err < 0)
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_warn(ice->card->dev,
|
|
|
|
"cannot initialize pro ac97, skipped\n");
|
2005-04-16 22:20:36 +00:00
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* I2S mixer only */
|
|
|
|
strcat(ice->card->mixername, "ICE1712 - multitrack");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static inline unsigned int eeprom_double(struct snd_ice1712 *ice, int idx)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
return (unsigned int)ice->eeprom.data[idx] | ((unsigned int)ice->eeprom.data[idx + 1] << 8);
|
|
|
|
}
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
static void snd_ice1712_proc_read(struct snd_info_entry *entry,
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_info_buffer *buffer)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = entry->private_data;
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned int idx;
|
|
|
|
|
|
|
|
snd_iprintf(buffer, "%s\n\n", ice->card->longname);
|
|
|
|
snd_iprintf(buffer, "EEPROM:\n");
|
|
|
|
|
|
|
|
snd_iprintf(buffer, " Subvendor : 0x%x\n", ice->eeprom.subvendor);
|
|
|
|
snd_iprintf(buffer, " Size : %i bytes\n", ice->eeprom.size);
|
|
|
|
snd_iprintf(buffer, " Version : %i\n", ice->eeprom.version);
|
|
|
|
snd_iprintf(buffer, " Codec : 0x%x\n", ice->eeprom.data[ICE_EEP1_CODEC]);
|
|
|
|
snd_iprintf(buffer, " ACLink : 0x%x\n", ice->eeprom.data[ICE_EEP1_ACLINK]);
|
|
|
|
snd_iprintf(buffer, " I2S ID : 0x%x\n", ice->eeprom.data[ICE_EEP1_I2SID]);
|
|
|
|
snd_iprintf(buffer, " S/PDIF : 0x%x\n", ice->eeprom.data[ICE_EEP1_SPDIF]);
|
|
|
|
snd_iprintf(buffer, " GPIO mask : 0x%x\n", ice->eeprom.gpiomask);
|
|
|
|
snd_iprintf(buffer, " GPIO state : 0x%x\n", ice->eeprom.gpiostate);
|
|
|
|
snd_iprintf(buffer, " GPIO direction : 0x%x\n", ice->eeprom.gpiodir);
|
|
|
|
snd_iprintf(buffer, " AC'97 main : 0x%x\n", eeprom_double(ice, ICE_EEP1_AC97_MAIN_LO));
|
|
|
|
snd_iprintf(buffer, " AC'97 pcm : 0x%x\n", eeprom_double(ice, ICE_EEP1_AC97_PCM_LO));
|
|
|
|
snd_iprintf(buffer, " AC'97 record : 0x%x\n", eeprom_double(ice, ICE_EEP1_AC97_REC_LO));
|
|
|
|
snd_iprintf(buffer, " AC'97 record src : 0x%x\n", ice->eeprom.data[ICE_EEP1_AC97_RECSRC]);
|
|
|
|
for (idx = 0; idx < 4; idx++)
|
|
|
|
snd_iprintf(buffer, " DAC ID #%i : 0x%x\n", idx, ice->eeprom.data[ICE_EEP1_DAC_ID + idx]);
|
|
|
|
for (idx = 0; idx < 4; idx++)
|
|
|
|
snd_iprintf(buffer, " ADC ID #%i : 0x%x\n", idx, ice->eeprom.data[ICE_EEP1_ADC_ID + idx]);
|
|
|
|
for (idx = 0x1c; idx < ice->eeprom.size; idx++)
|
|
|
|
snd_iprintf(buffer, " Extra #%02i : 0x%x\n", idx, ice->eeprom.data[idx]);
|
|
|
|
|
|
|
|
snd_iprintf(buffer, "\nRegisters:\n");
|
|
|
|
snd_iprintf(buffer, " PSDOUT03 : 0x%04x\n", (unsigned)inw(ICEMT(ice, ROUTE_PSDOUT03)));
|
|
|
|
snd_iprintf(buffer, " CAPTURE : 0x%08x\n", inl(ICEMT(ice, ROUTE_CAPTURE)));
|
|
|
|
snd_iprintf(buffer, " SPDOUT : 0x%04x\n", (unsigned)inw(ICEMT(ice, ROUTE_SPDOUT)));
|
|
|
|
snd_iprintf(buffer, " RATE : 0x%02x\n", (unsigned)inb(ICEMT(ice, RATE)));
|
2006-02-10 07:42:17 +00:00
|
|
|
snd_iprintf(buffer, " GPIO_DATA : 0x%02x\n", (unsigned)snd_ice1712_get_gpio_data(ice));
|
2008-09-07 10:17:02 +00:00
|
|
|
snd_iprintf(buffer, " GPIO_WRITE_MASK : 0x%02x\n", (unsigned)snd_ice1712_read(ice, ICE1712_IREG_GPIO_WRITE_MASK));
|
2006-02-10 07:42:17 +00:00
|
|
|
snd_iprintf(buffer, " GPIO_DIRECTION : 0x%02x\n", (unsigned)snd_ice1712_read(ice, ICE1712_IREG_GPIO_DIRECTION));
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static void snd_ice1712_proc_init(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2019-02-04 15:01:39 +00:00
|
|
|
snd_card_ro_proc_new(ice->card, "ice1712", ice, snd_ice1712_proc_read);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_eeprom_info(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_info *uinfo)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_BYTES;
|
2005-11-17 13:59:52 +00:00
|
|
|
uinfo->count = sizeof(struct snd_ice1712_eeprom);
|
2005-04-16 22:20:36 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_eeprom_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
memcpy(ucontrol->value.bytes.data, &ice->eeprom, sizeof(ice->eeprom));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_eeprom = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_CARD,
|
|
|
|
.name = "ICE1712 EEPROM",
|
|
|
|
.access = SNDRV_CTL_ELEM_ACCESS_READ,
|
|
|
|
.info = snd_ice1712_eeprom_info,
|
|
|
|
.get = snd_ice1712_eeprom_get
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
*/
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_spdif_info(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_info *uinfo)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_IEC958;
|
|
|
|
uinfo->count = 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_spdif_default_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ice->spdif.ops.default_get)
|
2008-09-07 10:17:02 +00:00
|
|
|
ice->spdif.ops.default_get(ice, ucontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_spdif_default_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ice->spdif.ops.default_put)
|
|
|
|
return ice->spdif.ops.default_put(ice, ucontrol);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_spdif_default =
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_PCM,
|
2008-09-07 10:17:02 +00:00
|
|
|
.name = SNDRV_CTL_NAME_IEC958("", PLAYBACK, DEFAULT),
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = snd_ice1712_spdif_info,
|
|
|
|
.get = snd_ice1712_spdif_default_get,
|
|
|
|
.put = snd_ice1712_spdif_default_put
|
|
|
|
};
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_spdif_maskc_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ice->spdif.ops.default_get) {
|
|
|
|
ucontrol->value.iec958.status[0] = IEC958_AES0_NONAUDIO |
|
|
|
|
IEC958_AES0_PROFESSIONAL |
|
|
|
|
IEC958_AES0_CON_NOT_COPYRIGHT |
|
|
|
|
IEC958_AES0_CON_EMPHASIS;
|
|
|
|
ucontrol->value.iec958.status[1] = IEC958_AES1_CON_ORIGINAL |
|
|
|
|
IEC958_AES1_CON_CATEGORY;
|
|
|
|
ucontrol->value.iec958.status[3] = IEC958_AES3_CON_FS;
|
|
|
|
} else {
|
|
|
|
ucontrol->value.iec958.status[0] = 0xff;
|
|
|
|
ucontrol->value.iec958.status[1] = 0xff;
|
|
|
|
ucontrol->value.iec958.status[2] = 0xff;
|
|
|
|
ucontrol->value.iec958.status[3] = 0xff;
|
|
|
|
ucontrol->value.iec958.status[4] = 0xff;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_spdif_maskp_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ice->spdif.ops.default_get) {
|
|
|
|
ucontrol->value.iec958.status[0] = IEC958_AES0_NONAUDIO |
|
|
|
|
IEC958_AES0_PROFESSIONAL |
|
|
|
|
IEC958_AES0_PRO_FS |
|
|
|
|
IEC958_AES0_PRO_EMPHASIS;
|
|
|
|
ucontrol->value.iec958.status[1] = IEC958_AES1_PRO_MODE;
|
|
|
|
} else {
|
|
|
|
ucontrol->value.iec958.status[0] = 0xff;
|
|
|
|
ucontrol->value.iec958.status[1] = 0xff;
|
|
|
|
ucontrol->value.iec958.status[2] = 0xff;
|
|
|
|
ucontrol->value.iec958.status[3] = 0xff;
|
|
|
|
ucontrol->value.iec958.status[4] = 0xff;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_spdif_maskc =
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
.access = SNDRV_CTL_ELEM_ACCESS_READ,
|
[ALSA] sound - fix .iface field of mixer control elements
Documentation,CS46xx driver,EMU10K1/EMU10K2 driver,AD1848 driver
SB16/AWE driver,CMIPCI driver,ENS1370/1+ driver,RME32 driver
RME96 driver,ICE1712 driver,ICE1724 driver,KORG1212 driver
RME HDSP driver,RME9652 driver
This patch changes .iface to SNDRV_CTL_ELEM_IFACE_MIXER whre _PCM or
_HWDEP was used in controls that are not associated with a specific PCM
(sub)stream or hwdep device, and changes some controls that got
inconsitent .iface values due to copy+paste errors. Furthermore, it
makes sure that all control that do use _PCM or _HWDEP use the correct
number in the .device field.
Signed-off-by: Clemens Ladisch <clemens@ladisch.de>
2005-07-29 13:32:58 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_PCM,
|
2008-09-07 10:17:02 +00:00
|
|
|
.name = SNDRV_CTL_NAME_IEC958("", PLAYBACK, CON_MASK),
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = snd_ice1712_spdif_info,
|
|
|
|
.get = snd_ice1712_spdif_maskc_get,
|
|
|
|
};
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_spdif_maskp =
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
.access = SNDRV_CTL_ELEM_ACCESS_READ,
|
[ALSA] sound - fix .iface field of mixer control elements
Documentation,CS46xx driver,EMU10K1/EMU10K2 driver,AD1848 driver
SB16/AWE driver,CMIPCI driver,ENS1370/1+ driver,RME32 driver
RME96 driver,ICE1712 driver,ICE1724 driver,KORG1212 driver
RME HDSP driver,RME9652 driver
This patch changes .iface to SNDRV_CTL_ELEM_IFACE_MIXER whre _PCM or
_HWDEP was used in controls that are not associated with a specific PCM
(sub)stream or hwdep device, and changes some controls that got
inconsitent .iface values due to copy+paste errors. Furthermore, it
makes sure that all control that do use _PCM or _HWDEP use the correct
number in the .device field.
Signed-off-by: Clemens Ladisch <clemens@ladisch.de>
2005-07-29 13:32:58 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_PCM,
|
2008-09-07 10:17:02 +00:00
|
|
|
.name = SNDRV_CTL_NAME_IEC958("", PLAYBACK, PRO_MASK),
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = snd_ice1712_spdif_info,
|
|
|
|
.get = snd_ice1712_spdif_maskp_get,
|
|
|
|
};
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_spdif_stream_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ice->spdif.ops.stream_get)
|
|
|
|
ice->spdif.ops.stream_get(ice, ucontrol);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_spdif_stream_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
if (ice->spdif.ops.stream_put)
|
|
|
|
return ice->spdif.ops.stream_put(ice, ucontrol);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_spdif_stream =
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
.access = (SNDRV_CTL_ELEM_ACCESS_READWRITE |
|
|
|
|
SNDRV_CTL_ELEM_ACCESS_INACTIVE),
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_PCM,
|
2008-09-07 10:17:02 +00:00
|
|
|
.name = SNDRV_CTL_NAME_IEC958("", PLAYBACK, PCM_STREAM),
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = snd_ice1712_spdif_info,
|
|
|
|
.get = snd_ice1712_spdif_stream_get,
|
|
|
|
.put = snd_ice1712_spdif_stream_put
|
|
|
|
};
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
int snd_ice1712_gpio_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned char mask = kcontrol->private_value & 0xff;
|
|
|
|
int invert = (kcontrol->private_value & (1<<24)) ? 1 : 0;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
snd_ice1712_save_gpio_status(ice);
|
2005-11-17 13:59:52 +00:00
|
|
|
ucontrol->value.integer.value[0] =
|
|
|
|
(snd_ice1712_gpio_read(ice) & mask ? 1 : 0) ^ invert;
|
2005-04-16 22:20:36 +00:00
|
|
|
snd_ice1712_restore_gpio_status(ice);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
int snd_ice1712_gpio_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned char mask = kcontrol->private_value & 0xff;
|
|
|
|
int invert = (kcontrol->private_value & (1<<24)) ? mask : 0;
|
|
|
|
unsigned int val, nval;
|
|
|
|
|
|
|
|
if (kcontrol->private_value & (1 << 31))
|
|
|
|
return -EPERM;
|
|
|
|
nval = (ucontrol->value.integer.value[0] ? mask : 0) ^ invert;
|
|
|
|
snd_ice1712_save_gpio_status(ice);
|
|
|
|
val = snd_ice1712_gpio_read(ice);
|
|
|
|
nval |= val & ~mask;
|
|
|
|
if (val != nval)
|
|
|
|
snd_ice1712_gpio_write(ice, nval);
|
|
|
|
snd_ice1712_restore_gpio_status(ice);
|
|
|
|
return val != nval;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* rate
|
|
|
|
*/
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_internal_clock_info(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_info *uinfo)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2007-01-29 14:26:36 +00:00
|
|
|
static const char * const texts[] = {
|
2005-04-16 22:20:36 +00:00
|
|
|
"8000", /* 0: 6 */
|
|
|
|
"9600", /* 1: 3 */
|
|
|
|
"11025", /* 2: 10 */
|
|
|
|
"12000", /* 3: 2 */
|
|
|
|
"16000", /* 4: 5 */
|
|
|
|
"22050", /* 5: 9 */
|
|
|
|
"24000", /* 6: 1 */
|
|
|
|
"32000", /* 7: 4 */
|
|
|
|
"44100", /* 8: 8 */
|
|
|
|
"48000", /* 9: 0 */
|
|
|
|
"64000", /* 10: 15 */
|
|
|
|
"88200", /* 11: 11 */
|
|
|
|
"96000", /* 12: 7 */
|
|
|
|
"IEC958 Input", /* 13: -- */
|
|
|
|
};
|
2014-10-20 16:18:33 +00:00
|
|
|
return snd_ctl_enum_info(uinfo, 1, 14, texts);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_internal_clock_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2007-01-29 14:26:36 +00:00
|
|
|
static const unsigned char xlate[16] = {
|
2005-04-16 22:20:36 +00:00
|
|
|
9, 6, 3, 1, 7, 4, 0, 12, 8, 5, 2, 11, 255, 255, 255, 10
|
|
|
|
};
|
|
|
|
unsigned char val;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
if (is_spdif_master(ice)) {
|
|
|
|
ucontrol->value.enumerated.item[0] = 13;
|
|
|
|
} else {
|
|
|
|
val = xlate[inb(ICEMT(ice, RATE)) & 15];
|
|
|
|
if (val == 255) {
|
|
|
|
snd_BUG();
|
|
|
|
val = 0;
|
|
|
|
}
|
|
|
|
ucontrol->value.enumerated.item[0] = val;
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_internal_clock_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2007-01-29 14:26:36 +00:00
|
|
|
static const unsigned int xrate[13] = {
|
2006-08-15 12:39:07 +00:00
|
|
|
8000, 9600, 11025, 12000, 16000, 22050, 24000,
|
2005-04-16 22:20:36 +00:00
|
|
|
32000, 44100, 48000, 64000, 88200, 96000
|
|
|
|
};
|
|
|
|
unsigned char oval;
|
|
|
|
int change = 0;
|
|
|
|
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
oval = inb(ICEMT(ice, RATE));
|
|
|
|
if (ucontrol->value.enumerated.item[0] == 13) {
|
|
|
|
outb(oval | ICE1712_SPDIF_MASTER, ICEMT(ice, RATE));
|
|
|
|
} else {
|
|
|
|
PRO_RATE_DEFAULT = xrate[ucontrol->value.integer.value[0] % 13];
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
snd_ice1712_set_pro_rate(ice, PRO_RATE_DEFAULT, 1);
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
}
|
|
|
|
change = inb(ICEMT(ice, RATE)) != oval;
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
if ((oval & ICE1712_SPDIF_MASTER) !=
|
2006-02-02 06:56:54 +00:00
|
|
|
(inb(ICEMT(ice, RATE)) & ICE1712_SPDIF_MASTER))
|
2008-09-07 10:17:02 +00:00
|
|
|
snd_ice1712_set_input_clock_source(ice, is_spdif_master(ice));
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_pro_internal_clock = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Multi Track Internal Clock",
|
|
|
|
.info = snd_ice1712_pro_internal_clock_info,
|
|
|
|
.get = snd_ice1712_pro_internal_clock_get,
|
|
|
|
.put = snd_ice1712_pro_internal_clock_put
|
|
|
|
};
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_internal_clock_default_info(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_info *uinfo)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2007-01-29 14:26:36 +00:00
|
|
|
static const char * const texts[] = {
|
2005-04-16 22:20:36 +00:00
|
|
|
"8000", /* 0: 6 */
|
|
|
|
"9600", /* 1: 3 */
|
|
|
|
"11025", /* 2: 10 */
|
|
|
|
"12000", /* 3: 2 */
|
|
|
|
"16000", /* 4: 5 */
|
|
|
|
"22050", /* 5: 9 */
|
|
|
|
"24000", /* 6: 1 */
|
|
|
|
"32000", /* 7: 4 */
|
|
|
|
"44100", /* 8: 8 */
|
|
|
|
"48000", /* 9: 0 */
|
|
|
|
"64000", /* 10: 15 */
|
|
|
|
"88200", /* 11: 11 */
|
|
|
|
"96000", /* 12: 7 */
|
2008-09-07 10:17:02 +00:00
|
|
|
/* "IEC958 Input", 13: -- */
|
2005-04-16 22:20:36 +00:00
|
|
|
};
|
2014-10-20 16:18:33 +00:00
|
|
|
return snd_ctl_enum_info(uinfo, 1, 13, texts);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_internal_clock_default_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
int val;
|
2007-01-29 14:26:36 +00:00
|
|
|
static const unsigned int xrate[13] = {
|
2006-08-15 12:39:07 +00:00
|
|
|
8000, 9600, 11025, 12000, 16000, 22050, 24000,
|
2005-04-16 22:20:36 +00:00
|
|
|
32000, 44100, 48000, 64000, 88200, 96000
|
|
|
|
};
|
|
|
|
|
|
|
|
for (val = 0; val < 13; val++) {
|
|
|
|
if (xrate[val] == PRO_RATE_DEFAULT)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ucontrol->value.enumerated.item[0] = val;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_internal_clock_default_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2007-01-29 14:26:36 +00:00
|
|
|
static const unsigned int xrate[13] = {
|
2006-08-15 12:39:07 +00:00
|
|
|
8000, 9600, 11025, 12000, 16000, 22050, 24000,
|
2005-04-16 22:20:36 +00:00
|
|
|
32000, 44100, 48000, 64000, 88200, 96000
|
|
|
|
};
|
|
|
|
unsigned char oval;
|
|
|
|
int change = 0;
|
|
|
|
|
|
|
|
oval = PRO_RATE_DEFAULT;
|
|
|
|
PRO_RATE_DEFAULT = xrate[ucontrol->value.integer.value[0] % 13];
|
|
|
|
change = PRO_RATE_DEFAULT != oval;
|
|
|
|
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_pro_internal_clock_default = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Multi Track Internal Clock Default",
|
|
|
|
.info = snd_ice1712_pro_internal_clock_default_info,
|
|
|
|
.get = snd_ice1712_pro_internal_clock_default_get,
|
|
|
|
.put = snd_ice1712_pro_internal_clock_default_put
|
|
|
|
};
|
|
|
|
|
2007-07-23 13:42:26 +00:00
|
|
|
#define snd_ice1712_pro_rate_locking_info snd_ctl_boolean_mono_info
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_rate_locking_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
ucontrol->value.integer.value[0] = PRO_RATE_LOCKED;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_rate_locking_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
int change = 0, nval;
|
|
|
|
|
|
|
|
nval = ucontrol->value.integer.value[0] ? 1 : 0;
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
change = PRO_RATE_LOCKED != nval;
|
|
|
|
PRO_RATE_LOCKED = nval;
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_pro_rate_locking = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Multi Track Rate Locking",
|
|
|
|
.info = snd_ice1712_pro_rate_locking_info,
|
|
|
|
.get = snd_ice1712_pro_rate_locking_get,
|
|
|
|
.put = snd_ice1712_pro_rate_locking_put
|
|
|
|
};
|
|
|
|
|
2007-07-23 13:42:26 +00:00
|
|
|
#define snd_ice1712_pro_rate_reset_info snd_ctl_boolean_mono_info
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_rate_reset_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
ucontrol->value.integer.value[0] = PRO_RATE_RESET;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_rate_reset_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
int change = 0, nval;
|
|
|
|
|
|
|
|
nval = ucontrol->value.integer.value[0] ? 1 : 0;
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
change = PRO_RATE_RESET != nval;
|
|
|
|
PRO_RATE_RESET = nval;
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_pro_rate_reset = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Multi Track Rate Reset",
|
|
|
|
.info = snd_ice1712_pro_rate_reset_info,
|
|
|
|
.get = snd_ice1712_pro_rate_reset_get,
|
|
|
|
.put = snd_ice1712_pro_rate_reset_put
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* routing
|
|
|
|
*/
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_route_info(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_info *uinfo)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2007-01-29 14:26:36 +00:00
|
|
|
static const char * const texts[] = {
|
2005-04-16 22:20:36 +00:00
|
|
|
"PCM Out", /* 0 */
|
|
|
|
"H/W In 0", "H/W In 1", "H/W In 2", "H/W In 3", /* 1-4 */
|
|
|
|
"H/W In 4", "H/W In 5", "H/W In 6", "H/W In 7", /* 5-8 */
|
|
|
|
"IEC958 In L", "IEC958 In R", /* 9-10 */
|
|
|
|
"Digital Mixer", /* 11 - optional */
|
|
|
|
};
|
2014-10-20 16:18:33 +00:00
|
|
|
int num_items = snd_ctl_get_ioffidx(kcontrol, &uinfo->id) < 2 ? 12 : 11;
|
|
|
|
return snd_ctl_enum_info(uinfo, 1, num_items, texts);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_route_analog_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
|
|
|
|
unsigned int val, cval;
|
|
|
|
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
val = inw(ICEMT(ice, ROUTE_PSDOUT03));
|
|
|
|
cval = inl(ICEMT(ice, ROUTE_CAPTURE));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
|
|
|
|
val >>= ((idx % 2) * 8) + ((idx / 2) * 2);
|
|
|
|
val &= 3;
|
|
|
|
cval >>= ((idx / 2) * 8) + ((idx % 2) * 4);
|
|
|
|
if (val == 1 && idx < 2)
|
|
|
|
ucontrol->value.enumerated.item[0] = 11;
|
|
|
|
else if (val == 2)
|
|
|
|
ucontrol->value.enumerated.item[0] = (cval & 7) + 1;
|
|
|
|
else if (val == 3)
|
|
|
|
ucontrol->value.enumerated.item[0] = ((cval >> 3) & 1) + 9;
|
|
|
|
else
|
|
|
|
ucontrol->value.enumerated.item[0] = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_route_analog_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
int change, shift;
|
|
|
|
int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
|
|
|
|
unsigned int val, old_val, nval;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
/* update PSDOUT */
|
|
|
|
if (ucontrol->value.enumerated.item[0] >= 11)
|
|
|
|
nval = idx < 2 ? 1 : 0; /* dig mixer (or pcm) */
|
|
|
|
else if (ucontrol->value.enumerated.item[0] >= 9)
|
|
|
|
nval = 3; /* spdif in */
|
|
|
|
else if (ucontrol->value.enumerated.item[0] >= 1)
|
|
|
|
nval = 2; /* analog in */
|
|
|
|
else
|
|
|
|
nval = 0; /* pcm */
|
|
|
|
shift = ((idx % 2) * 8) + ((idx / 2) * 2);
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
val = old_val = inw(ICEMT(ice, ROUTE_PSDOUT03));
|
|
|
|
val &= ~(0x03 << shift);
|
|
|
|
val |= nval << shift;
|
|
|
|
change = val != old_val;
|
|
|
|
if (change)
|
|
|
|
outw(val, ICEMT(ice, ROUTE_PSDOUT03));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
if (nval < 2) /* dig mixer of pcm */
|
|
|
|
return change;
|
|
|
|
|
|
|
|
/* update CAPTURE */
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
val = old_val = inl(ICEMT(ice, ROUTE_CAPTURE));
|
|
|
|
shift = ((idx / 2) * 8) + ((idx % 2) * 4);
|
|
|
|
if (nval == 2) { /* analog in */
|
|
|
|
nval = ucontrol->value.enumerated.item[0] - 1;
|
|
|
|
val &= ~(0x07 << shift);
|
|
|
|
val |= nval << shift;
|
|
|
|
} else { /* spdif in */
|
|
|
|
nval = (ucontrol->value.enumerated.item[0] - 9) << 3;
|
|
|
|
val &= ~(0x08 << shift);
|
|
|
|
val |= nval << shift;
|
|
|
|
}
|
|
|
|
if (val != old_val) {
|
|
|
|
change = 1;
|
|
|
|
outl(val, ICEMT(ice, ROUTE_CAPTURE));
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_route_spdif_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
|
|
|
|
unsigned int val, cval;
|
|
|
|
val = inw(ICEMT(ice, ROUTE_SPDOUT));
|
|
|
|
cval = (val >> (idx * 4 + 8)) & 0x0f;
|
|
|
|
val = (val >> (idx * 2)) & 0x03;
|
|
|
|
if (val == 1)
|
|
|
|
ucontrol->value.enumerated.item[0] = 11;
|
|
|
|
else if (val == 2)
|
|
|
|
ucontrol->value.enumerated.item[0] = (cval & 7) + 1;
|
|
|
|
else if (val == 3)
|
|
|
|
ucontrol->value.enumerated.item[0] = ((cval >> 3) & 1) + 9;
|
|
|
|
else
|
|
|
|
ucontrol->value.enumerated.item[0] = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_route_spdif_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
int change, shift;
|
|
|
|
int idx = snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);
|
|
|
|
unsigned int val, old_val, nval;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
/* update SPDOUT */
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
val = old_val = inw(ICEMT(ice, ROUTE_SPDOUT));
|
|
|
|
if (ucontrol->value.enumerated.item[0] >= 11)
|
|
|
|
nval = 1;
|
|
|
|
else if (ucontrol->value.enumerated.item[0] >= 9)
|
|
|
|
nval = 3;
|
|
|
|
else if (ucontrol->value.enumerated.item[0] >= 1)
|
|
|
|
nval = 2;
|
|
|
|
else
|
|
|
|
nval = 0;
|
|
|
|
shift = idx * 2;
|
|
|
|
val &= ~(0x03 << shift);
|
|
|
|
val |= nval << shift;
|
|
|
|
shift = idx * 4 + 8;
|
|
|
|
if (nval == 2) {
|
|
|
|
nval = ucontrol->value.enumerated.item[0] - 1;
|
|
|
|
val &= ~(0x07 << shift);
|
|
|
|
val |= nval << shift;
|
|
|
|
} else if (nval == 3) {
|
|
|
|
nval = (ucontrol->value.enumerated.item[0] - 9) << 3;
|
|
|
|
val &= ~(0x08 << shift);
|
|
|
|
val |= nval << shift;
|
|
|
|
}
|
|
|
|
change = val != old_val;
|
|
|
|
if (change)
|
|
|
|
outw(val, ICEMT(ice, ROUTE_SPDOUT));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2017-05-17 11:50:46 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_mixer_pro_analog_route = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "H/W Playback Route",
|
|
|
|
.info = snd_ice1712_pro_route_info,
|
|
|
|
.get = snd_ice1712_pro_route_analog_get,
|
|
|
|
.put = snd_ice1712_pro_route_analog_put,
|
|
|
|
};
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_mixer_pro_spdif_route = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
2008-09-07 10:17:02 +00:00
|
|
|
.name = SNDRV_CTL_NAME_IEC958("", PLAYBACK, NONE) "Route",
|
2005-04-16 22:20:36 +00:00
|
|
|
.info = snd_ice1712_pro_route_info,
|
|
|
|
.get = snd_ice1712_pro_route_spdif_get,
|
|
|
|
.put = snd_ice1712_pro_route_spdif_put,
|
|
|
|
.count = 2,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_volume_rate_info(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_info *uinfo)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
|
|
|
|
uinfo->count = 1;
|
|
|
|
uinfo->value.integer.min = 0;
|
|
|
|
uinfo->value.integer.max = 255;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_volume_rate_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
ucontrol->value.integer.value[0] = inb(ICEMT(ice, MONITOR_RATE));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_volume_rate_put(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
int change;
|
|
|
|
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
change = inb(ICEMT(ice, MONITOR_RATE)) != ucontrol->value.integer.value[0];
|
|
|
|
outb(ucontrol->value.integer.value[0], ICEMT(ice, MONITOR_RATE));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return change;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_mixer_pro_volume_rate = {
|
2005-04-16 22:20:36 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
|
|
|
|
.name = "Multi Track Volume Rate",
|
|
|
|
.info = snd_ice1712_pro_volume_rate_info,
|
|
|
|
.get = snd_ice1712_pro_volume_rate_get,
|
|
|
|
.put = snd_ice1712_pro_volume_rate_put
|
|
|
|
};
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_peak_info(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_info *uinfo)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
|
|
|
|
uinfo->count = 22;
|
|
|
|
uinfo->value.integer.min = 0;
|
|
|
|
uinfo->value.integer.max = 255;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-11-17 13:59:52 +00:00
|
|
|
static int snd_ice1712_pro_peak_get(struct snd_kcontrol *kcontrol,
|
|
|
|
struct snd_ctl_elem_value *ucontrol)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_ice1712 *ice = snd_kcontrol_chip(kcontrol);
|
2005-04-16 22:20:36 +00:00
|
|
|
int idx;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
for (idx = 0; idx < 22; idx++) {
|
|
|
|
outb(idx, ICEMT(ice, MONITOR_PEAKINDEX));
|
|
|
|
ucontrol->value.integer.value[idx] = inb(ICEMT(ice, MONITOR_PEAKDATA));
|
|
|
|
}
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-19 18:48:09 +00:00
|
|
|
static const struct snd_kcontrol_new snd_ice1712_mixer_pro_peak = {
|
2009-10-06 14:04:11 +00:00
|
|
|
.iface = SNDRV_CTL_ELEM_IFACE_PCM,
|
2005-04-16 22:20:36 +00:00
|
|
|
.name = "Multi Track Peak",
|
|
|
|
.access = SNDRV_CTL_ELEM_ACCESS_READ | SNDRV_CTL_ELEM_ACCESS_VOLATILE,
|
|
|
|
.info = snd_ice1712_pro_peak_info,
|
|
|
|
.get = snd_ice1712_pro_peak_get
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* list of available boards
|
|
|
|
*/
|
2020-01-03 08:17:04 +00:00
|
|
|
static const struct snd_ice1712_card_info *card_tables[] = {
|
2005-04-16 22:20:36 +00:00
|
|
|
snd_ice1712_hoontech_cards,
|
|
|
|
snd_ice1712_delta_cards,
|
|
|
|
snd_ice1712_ews_cards,
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static unsigned char snd_ice1712_read_i2c(struct snd_ice1712 *ice,
|
|
|
|
unsigned char dev,
|
|
|
|
unsigned char addr)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
long t = 0x10000;
|
|
|
|
|
|
|
|
outb(addr, ICEREG(ice, I2C_BYTE_ADDR));
|
|
|
|
outb(dev & ~ICE1712_I2C_WRITE, ICEREG(ice, I2C_DEV_ADDR));
|
|
|
|
while (t-- > 0 && (inb(ICEREG(ice, I2C_CTRL)) & ICE1712_I2C_BUSY)) ;
|
|
|
|
return inb(ICEREG(ice, I2C_DATA));
|
|
|
|
}
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static int snd_ice1712_read_eeprom(struct snd_ice1712 *ice,
|
|
|
|
const char *modelname)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2017-11-05 20:11:01 +00:00
|
|
|
int dev = ICE_I2C_EEPROM_ADDR; /* I2C EEPROM device address */
|
2005-04-16 22:20:36 +00:00
|
|
|
unsigned int i, size;
|
2020-01-03 08:17:04 +00:00
|
|
|
const struct snd_ice1712_card_info * const *tbl, *c;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
if (!modelname || !*modelname) {
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->eeprom.subvendor = 0;
|
|
|
|
if ((inb(ICEREG(ice, I2C_CTRL)) & ICE1712_I2C_EEPROM) != 0)
|
|
|
|
ice->eeprom.subvendor = (snd_ice1712_read_i2c(ice, dev, 0x00) << 0) |
|
2008-09-07 10:17:02 +00:00
|
|
|
(snd_ice1712_read_i2c(ice, dev, 0x01) << 8) |
|
|
|
|
(snd_ice1712_read_i2c(ice, dev, 0x02) << 16) |
|
2005-04-16 22:20:36 +00:00
|
|
|
(snd_ice1712_read_i2c(ice, dev, 0x03) << 24);
|
2005-11-17 13:59:52 +00:00
|
|
|
if (ice->eeprom.subvendor == 0 ||
|
|
|
|
ice->eeprom.subvendor == (unsigned int)-1) {
|
2005-04-16 22:20:36 +00:00
|
|
|
/* invalid subvendor from EEPROM, try the PCI subststem ID instead */
|
|
|
|
u16 vendor, device;
|
|
|
|
pci_read_config_word(ice->pci, PCI_SUBSYSTEM_VENDOR_ID, &vendor);
|
|
|
|
pci_read_config_word(ice->pci, PCI_SUBSYSTEM_ID, &device);
|
|
|
|
ice->eeprom.subvendor = ((unsigned int)swab16(vendor) << 16) | swab16(device);
|
|
|
|
if (ice->eeprom.subvendor == 0 || ice->eeprom.subvendor == (unsigned int)-1) {
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_err(ice->card->dev,
|
|
|
|
"No valid ID is found\n");
|
2005-04-16 22:20:36 +00:00
|
|
|
return -ENXIO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (tbl = card_tables; *tbl; tbl++) {
|
|
|
|
for (c = *tbl; c->subvendor; c++) {
|
2008-09-07 10:17:02 +00:00
|
|
|
if (modelname && c->model && !strcmp(modelname, c->model)) {
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_info(ice->card->dev,
|
|
|
|
"Using board model %s\n", c->name);
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->eeprom.subvendor = c->subvendor;
|
|
|
|
} else if (c->subvendor != ice->eeprom.subvendor)
|
|
|
|
continue;
|
2008-09-07 10:17:02 +00:00
|
|
|
if (!c->eeprom_size || !c->eeprom_data)
|
2005-04-16 22:20:36 +00:00
|
|
|
goto found;
|
|
|
|
/* if the EEPROM is given by the driver, use it */
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_dbg(ice->card->dev, "using the defined eeprom..\n");
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->eeprom.version = 1;
|
|
|
|
ice->eeprom.size = c->eeprom_size + 6;
|
|
|
|
memcpy(ice->eeprom.data, c->eeprom_data, c->eeprom_size);
|
|
|
|
goto read_skipped;
|
|
|
|
}
|
|
|
|
}
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_warn(ice->card->dev, "No matching model found for ID 0x%x\n",
|
2005-11-17 13:59:52 +00:00
|
|
|
ice->eeprom.subvendor);
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
found:
|
|
|
|
ice->eeprom.size = snd_ice1712_read_i2c(ice, dev, 0x04);
|
|
|
|
if (ice->eeprom.size < 6)
|
|
|
|
ice->eeprom.size = 32; /* FIXME: any cards without the correct size? */
|
|
|
|
else if (ice->eeprom.size > 32) {
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_err(ice->card->dev,
|
|
|
|
"invalid EEPROM (size = %i)\n", ice->eeprom.size);
|
2005-04-16 22:20:36 +00:00
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
ice->eeprom.version = snd_ice1712_read_i2c(ice, dev, 0x05);
|
|
|
|
if (ice->eeprom.version != 1) {
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_err(ice->card->dev, "invalid EEPROM version %i\n",
|
2005-11-17 13:59:52 +00:00
|
|
|
ice->eeprom.version);
|
2005-04-16 22:20:36 +00:00
|
|
|
/* return -EIO; */
|
|
|
|
}
|
|
|
|
size = ice->eeprom.size - 6;
|
|
|
|
for (i = 0; i < size; i++)
|
|
|
|
ice->eeprom.data[i] = snd_ice1712_read_i2c(ice, dev, i + 6);
|
|
|
|
|
|
|
|
read_skipped:
|
|
|
|
ice->eeprom.gpiomask = ice->eeprom.data[ICE_EEP1_GPIO_MASK];
|
|
|
|
ice->eeprom.gpiostate = ice->eeprom.data[ICE_EEP1_GPIO_STATE];
|
|
|
|
ice->eeprom.gpiodir = ice->eeprom.data[ICE_EEP1_GPIO_DIR];
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static int snd_ice1712_chip_init(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
outb(ICE1712_RESET | ICE1712_NATIVE, ICEREG(ice, CONTROL));
|
|
|
|
udelay(200);
|
|
|
|
outb(ICE1712_NATIVE, ICEREG(ice, CONTROL));
|
|
|
|
udelay(200);
|
2006-05-23 11:29:51 +00:00
|
|
|
if (ice->eeprom.subvendor == ICE1712_SUBDEVICE_DMX6FIRE &&
|
|
|
|
!ice->dxr_enable)
|
|
|
|
/* Set eeprom value to limit active ADCs and DACs to 6;
|
|
|
|
* Also disable AC97 as no hardware in standard 6fire card/box
|
|
|
|
* Note: DXR extensions are not currently supported
|
|
|
|
*/
|
|
|
|
ice->eeprom.data[ICE_EEP1_CODEC] = 0x3a;
|
|
|
|
pci_write_config_byte(ice->pci, 0x60, ice->eeprom.data[ICE_EEP1_CODEC]);
|
2005-04-16 22:20:36 +00:00
|
|
|
pci_write_config_byte(ice->pci, 0x61, ice->eeprom.data[ICE_EEP1_ACLINK]);
|
|
|
|
pci_write_config_byte(ice->pci, 0x62, ice->eeprom.data[ICE_EEP1_I2SID]);
|
|
|
|
pci_write_config_byte(ice->pci, 0x63, ice->eeprom.data[ICE_EEP1_SPDIF]);
|
2020-05-18 17:57:28 +00:00
|
|
|
if (ice->eeprom.subvendor != ICE1712_SUBDEVICE_STDSP24 &&
|
|
|
|
ice->eeprom.subvendor != ICE1712_SUBDEVICE_STAUDIO_ADCIII) {
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->gpio.write_mask = ice->eeprom.gpiomask;
|
|
|
|
ice->gpio.direction = ice->eeprom.gpiodir;
|
2005-11-17 13:59:52 +00:00
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_WRITE_MASK,
|
|
|
|
ice->eeprom.gpiomask);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_DIRECTION,
|
|
|
|
ice->eeprom.gpiodir);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_DATA,
|
|
|
|
ice->eeprom.gpiostate);
|
2005-04-16 22:20:36 +00:00
|
|
|
} else {
|
|
|
|
ice->gpio.write_mask = 0xc0;
|
|
|
|
ice->gpio.direction = 0xff;
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_WRITE_MASK, 0xc0);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_DIRECTION, 0xff);
|
2005-11-17 13:59:52 +00:00
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_GPIO_DATA,
|
|
|
|
ICE1712_STDSP24_CLOCK_BIT);
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_PRO_POWERDOWN, 0);
|
|
|
|
if (!(ice->eeprom.data[ICE_EEP1_CODEC] & ICE1712_CFG_NO_CON_AC97)) {
|
|
|
|
outb(ICE1712_AC97_WARM, ICEREG(ice, AC97_CMD));
|
|
|
|
udelay(100);
|
|
|
|
outb(0, ICEREG(ice, AC97_CMD));
|
|
|
|
udelay(200);
|
|
|
|
snd_ice1712_write(ice, ICE1712_IREG_CONSUMER_POWERDOWN, 0);
|
|
|
|
}
|
|
|
|
snd_ice1712_set_pro_rate(ice, 48000, 1);
|
2014-03-30 21:37:30 +00:00
|
|
|
/* unmask used interrupts */
|
|
|
|
outb(((ice->eeprom.data[ICE_EEP1_CODEC] & ICE1712_CFG_2xMPU401) == 0 ?
|
|
|
|
ICE1712_IRQ_MPU2 : 0) |
|
|
|
|
((ice->eeprom.data[ICE_EEP1_CODEC] & ICE1712_CFG_NO_CON_AC97) ?
|
|
|
|
ICE1712_IRQ_PBKDS | ICE1712_IRQ_CONCAP | ICE1712_IRQ_CONPBK : 0),
|
|
|
|
ICEREG(ice, IRQMASK));
|
|
|
|
outb(0x00, ICEMT(ice, IRQ));
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
int snd_ice1712_spdif_build_controls(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
int err;
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_kcontrol *kctl;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2008-08-08 15:12:14 +00:00
|
|
|
if (snd_BUG_ON(!ice->pcm_pro))
|
|
|
|
return -EIO;
|
2005-04-16 22:20:36 +00:00
|
|
|
err = snd_ctl_add(ice->card, kctl = snd_ctl_new1(&snd_ice1712_spdif_default, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
kctl->id.device = ice->pcm_pro->device;
|
|
|
|
err = snd_ctl_add(ice->card, kctl = snd_ctl_new1(&snd_ice1712_spdif_maskc, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
kctl->id.device = ice->pcm_pro->device;
|
|
|
|
err = snd_ctl_add(ice->card, kctl = snd_ctl_new1(&snd_ice1712_spdif_maskp, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
kctl->id.device = ice->pcm_pro->device;
|
|
|
|
err = snd_ctl_add(ice->card, kctl = snd_ctl_new1(&snd_ice1712_spdif_stream, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
kctl->id.device = ice->pcm_pro->device;
|
|
|
|
ice->spdif.stream_ctl = kctl;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static int snd_ice1712_build_controls(struct snd_ice1712 *ice)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
int err;
|
|
|
|
|
|
|
|
err = snd_ctl_add(ice->card, snd_ctl_new1(&snd_ice1712_eeprom, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
err = snd_ctl_add(ice->card, snd_ctl_new1(&snd_ice1712_pro_internal_clock, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
err = snd_ctl_add(ice->card, snd_ctl_new1(&snd_ice1712_pro_internal_clock_default, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = snd_ctl_add(ice->card, snd_ctl_new1(&snd_ice1712_pro_rate_locking, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
err = snd_ctl_add(ice->card, snd_ctl_new1(&snd_ice1712_pro_rate_reset, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
if (ice->num_total_dacs > 0) {
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_kcontrol_new tmp = snd_ice1712_mixer_pro_analog_route;
|
2005-04-16 22:20:36 +00:00
|
|
|
tmp.count = ice->num_total_dacs;
|
|
|
|
err = snd_ctl_add(ice->card, snd_ctl_new1(&tmp, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
err = snd_ctl_add(ice->card, snd_ctl_new1(&snd_ice1712_mixer_pro_spdif_route, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
|
|
|
|
|
|
|
err = snd_ctl_add(ice->card, snd_ctl_new1(&snd_ice1712_mixer_pro_volume_rate, ice));
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
2014-11-14 10:39:05 +00:00
|
|
|
return snd_ctl_add(ice->card,
|
|
|
|
snd_ctl_new1(&snd_ice1712_mixer_pro_peak, ice));
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2021-07-15 07:58:57 +00:00
|
|
|
static void snd_ice1712_free(struct snd_card *card)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2021-07-15 07:58:57 +00:00
|
|
|
struct snd_ice1712 *ice = card->private_data;
|
|
|
|
|
|
|
|
if (ice->card_info && ice->card_info->chip_exit)
|
|
|
|
ice->card_info->chip_exit(ice);
|
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
/* mask all interrupts */
|
2014-08-27 04:21:48 +00:00
|
|
|
outb(ICE1712_MULTI_CAPTURE | ICE1712_MULTI_PLAYBACK, ICEMT(ice, IRQ));
|
2005-04-16 22:20:36 +00:00
|
|
|
outb(0xff, ICEREG(ice, IRQMASK));
|
2008-04-22 11:50:34 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
snd_ice1712_akm4xxx_free(ice);
|
|
|
|
}
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static int snd_ice1712_create(struct snd_card *card,
|
|
|
|
struct pci_dev *pci,
|
|
|
|
const char *modelname,
|
|
|
|
int omni,
|
|
|
|
int cs8427_timeout,
|
2021-07-15 07:58:57 +00:00
|
|
|
int dxr_enable)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
2021-07-15 07:58:57 +00:00
|
|
|
struct snd_ice1712 *ice = card->private_data;
|
2005-04-16 22:20:36 +00:00
|
|
|
int err;
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
/* enable PCI device */
|
2021-07-15 07:58:57 +00:00
|
|
|
err = pcim_enable_device(pci);
|
2008-09-07 10:17:02 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
/* check, if we can restrict PCI DMA transfers to 28 bits */
|
2021-01-14 12:54:11 +00:00
|
|
|
if (dma_set_mask_and_coherent(&pci->dev, DMA_BIT_MASK(28))) {
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_err(card->dev,
|
|
|
|
"architecture does not support 28bit PCI busmaster DMA\n");
|
2005-04-16 22:20:36 +00:00
|
|
|
return -ENXIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
ice->omni = omni ? 1 : 0;
|
|
|
|
if (cs8427_timeout < 1)
|
|
|
|
cs8427_timeout = 1;
|
|
|
|
else if (cs8427_timeout > 1000)
|
|
|
|
cs8427_timeout = 1000;
|
|
|
|
ice->cs8427_timeout = cs8427_timeout;
|
2006-02-08 06:40:33 +00:00
|
|
|
ice->dxr_enable = dxr_enable;
|
2005-04-16 22:20:36 +00:00
|
|
|
spin_lock_init(&ice->reg_lock);
|
2006-01-16 15:34:20 +00:00
|
|
|
mutex_init(&ice->gpio_mutex);
|
|
|
|
mutex_init(&ice->i2c_mutex);
|
|
|
|
mutex_init(&ice->open_mutex);
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->gpio.set_mask = snd_ice1712_set_gpio_mask;
|
2009-09-16 20:25:38 +00:00
|
|
|
ice->gpio.get_mask = snd_ice1712_get_gpio_mask;
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->gpio.set_dir = snd_ice1712_set_gpio_dir;
|
2009-09-16 20:25:38 +00:00
|
|
|
ice->gpio.get_dir = snd_ice1712_get_gpio_dir;
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->gpio.set_data = snd_ice1712_set_gpio_data;
|
|
|
|
ice->gpio.get_data = snd_ice1712_get_gpio_data;
|
|
|
|
|
|
|
|
ice->spdif.cs8403_bits =
|
|
|
|
ice->spdif.cs8403_stream_bits = (0x01 | /* consumer format */
|
|
|
|
0x10 | /* no emphasis */
|
|
|
|
0x20); /* PCM encoder/decoder */
|
|
|
|
ice->card = card;
|
|
|
|
ice->pci = pci;
|
|
|
|
ice->irq = -1;
|
|
|
|
pci_set_master(pci);
|
2014-03-30 21:37:30 +00:00
|
|
|
/* disable legacy emulation */
|
2005-04-16 22:20:36 +00:00
|
|
|
pci_write_config_word(ice->pci, 0x40, 0x807f);
|
|
|
|
pci_write_config_word(ice->pci, 0x42, 0x0006);
|
|
|
|
snd_ice1712_proc_init(ice);
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
err = pci_request_regions(pci, "ICE1712");
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
ice->port = pci_resource_start(pci, 0);
|
|
|
|
ice->ddma_port = pci_resource_start(pci, 1);
|
|
|
|
ice->dmapath_port = pci_resource_start(pci, 2);
|
|
|
|
ice->profi_port = pci_resource_start(pci, 3);
|
|
|
|
|
2021-07-15 07:58:57 +00:00
|
|
|
if (devm_request_irq(&pci->dev, pci->irq, snd_ice1712_interrupt,
|
|
|
|
IRQF_SHARED, KBUILD_MODNAME, ice)) {
|
2014-02-25 16:16:16 +00:00
|
|
|
dev_err(card->dev, "unable to grab IRQ %d\n", pci->irq);
|
2005-04-16 22:20:36 +00:00
|
|
|
return -EIO;
|
|
|
|
}
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
ice->irq = pci->irq;
|
2019-12-10 06:34:21 +00:00
|
|
|
card->sync_irq = ice->irq;
|
2021-07-15 07:58:57 +00:00
|
|
|
card->private_free = snd_ice1712_free;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2021-07-15 07:58:57 +00:00
|
|
|
if (snd_ice1712_read_eeprom(ice, modelname) < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return -EIO;
|
2021-07-15 07:58:57 +00:00
|
|
|
if (snd_ice1712_chip_init(ice) < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Registration
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static struct snd_ice1712_card_info no_matched;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2012-12-06 17:35:10 +00:00
|
|
|
static int snd_ice1712_probe(struct pci_dev *pci,
|
|
|
|
const struct pci_device_id *pci_id)
|
2005-04-16 22:20:36 +00:00
|
|
|
{
|
|
|
|
static int dev;
|
2005-11-17 13:59:52 +00:00
|
|
|
struct snd_card *card;
|
|
|
|
struct snd_ice1712 *ice;
|
2005-04-16 22:20:36 +00:00
|
|
|
int pcm_dev = 0, err;
|
2020-01-03 08:17:04 +00:00
|
|
|
const struct snd_ice1712_card_info * const *tbl, *c;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
if (dev >= SNDRV_CARDS)
|
|
|
|
return -ENODEV;
|
|
|
|
if (!enable[dev]) {
|
|
|
|
dev++;
|
|
|
|
return -ENOENT;
|
|
|
|
}
|
|
|
|
|
2014-01-29 13:20:19 +00:00
|
|
|
err = snd_card_new(&pci->dev, index[dev], id[dev], THIS_MODULE,
|
2021-07-15 07:58:57 +00:00
|
|
|
sizeof(*ice), &card);
|
2008-12-28 15:44:30 +00:00
|
|
|
if (err < 0)
|
|
|
|
return err;
|
2021-07-15 07:58:57 +00:00
|
|
|
ice = card->private_data;
|
2005-04-16 22:20:36 +00:00
|
|
|
|
|
|
|
strcpy(card->driver, "ICE1712");
|
|
|
|
strcpy(card->shortname, "ICEnsemble ICE1712");
|
2008-09-07 10:17:02 +00:00
|
|
|
|
|
|
|
err = snd_ice1712_create(card, pci, model[dev], omni[dev],
|
2021-07-15 07:58:57 +00:00
|
|
|
cs8427_timeout[dev], dxr_enable[dev]);
|
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
|
|
|
|
for (tbl = card_tables; *tbl; tbl++) {
|
|
|
|
for (c = *tbl; c->subvendor; c++) {
|
|
|
|
if (c->subvendor == ice->eeprom.subvendor) {
|
|
|
|
strcpy(card->shortname, c->name);
|
|
|
|
if (c->driver) /* specific driver? */
|
|
|
|
strcpy(card->driver, c->driver);
|
|
|
|
if (c->chip_init) {
|
2008-09-07 10:17:02 +00:00
|
|
|
err = c->chip_init(ice);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
}
|
2021-07-15 07:58:57 +00:00
|
|
|
ice->card_info = c;
|
2005-04-16 22:20:36 +00:00
|
|
|
goto __found;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c = &no_matched;
|
|
|
|
__found:
|
|
|
|
|
2015-01-02 11:24:51 +00:00
|
|
|
err = snd_ice1712_pcm_profi(ice, pcm_dev++);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2008-10-30 11:09:55 +00:00
|
|
|
if (ice_has_con_ac97(ice)) {
|
2015-01-02 11:24:51 +00:00
|
|
|
err = snd_ice1712_pcm(ice, pcm_dev++);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
2008-10-30 11:09:55 +00:00
|
|
|
}
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
err = snd_ice1712_ac97_mixer(ice);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
err = snd_ice1712_build_controls(ice);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
|
|
|
|
if (c->build_controls) {
|
2008-09-07 10:17:02 +00:00
|
|
|
err = c->build_controls(ice);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2008-10-30 11:09:55 +00:00
|
|
|
if (ice_has_con_ac97(ice)) {
|
2015-01-02 11:24:51 +00:00
|
|
|
err = snd_ice1712_pcm_ds(ice, pcm_dev++);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
2008-10-30 11:09:55 +00:00
|
|
|
}
|
2005-04-16 22:20:36 +00:00
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
if (!c->no_mpu401) {
|
|
|
|
err = snd_mpu401_uart_new(card, 0, MPU401_HW_ICE1712,
|
|
|
|
ICEREG(ice, MPU1_CTRL),
|
ALSA: mpu401: clean up interrupt specification
The semantics of snd_mpu401_uart_new()'s interrupt parameters are
somewhat counterintuitive: To prevent the function from allocating its
own interrupt, either the irq number must be invalid, or the irq_flags
parameter must be zero. At the same time, the irq parameter being
invalid specifies that the mpu401 code has to work without an interrupt
allocated by the caller. This implies that, if there is an interrupt
and it is allocated by the caller, the irq parameter must be set to
a valid-looking number which then isn't actually used.
With the removal of IRQF_DISABLED, zero becomes a valid irq_flags value,
which forces us to handle the parameters differently.
This patch introduces a new flag MPU401_INFO_IRQ_HOOK for when the
device interrupt is handled by the caller, and makes the allocation of
the interrupt to depend only on the irq parameter. As suggested by
Takashi, the irq_flags parameter was dropped because, when used, it had
the constant value IRQF_DISABLED.
Signed-off-by: Clemens Ladisch <clemens@ladisch.de>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2011-09-13 09:24:41 +00:00
|
|
|
c->mpu401_1_info_flags |
|
|
|
|
MPU401_INFO_INTEGRATED | MPU401_INFO_IRQ_HOOK,
|
|
|
|
-1, &ice->rmidi[0]);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
2006-04-26 16:13:59 +00:00
|
|
|
if (c->mpu401_1_name)
|
2011-03-31 01:57:33 +00:00
|
|
|
/* Preferred name available in card_info */
|
2006-04-26 16:13:59 +00:00
|
|
|
snprintf(ice->rmidi[0]->name,
|
|
|
|
sizeof(ice->rmidi[0]->name),
|
|
|
|
"%s %d", c->mpu401_1_name, card->number);
|
|
|
|
|
|
|
|
if (ice->eeprom.data[ICE_EEP1_CODEC] & ICE1712_CFG_2xMPU401) {
|
|
|
|
/* 2nd port used */
|
2008-09-07 10:17:02 +00:00
|
|
|
err = snd_mpu401_uart_new(card, 1, MPU401_HW_ICE1712,
|
|
|
|
ICEREG(ice, MPU2_CTRL),
|
ALSA: mpu401: clean up interrupt specification
The semantics of snd_mpu401_uart_new()'s interrupt parameters are
somewhat counterintuitive: To prevent the function from allocating its
own interrupt, either the irq number must be invalid, or the irq_flags
parameter must be zero. At the same time, the irq parameter being
invalid specifies that the mpu401 code has to work without an interrupt
allocated by the caller. This implies that, if there is an interrupt
and it is allocated by the caller, the irq parameter must be set to
a valid-looking number which then isn't actually used.
With the removal of IRQF_DISABLED, zero becomes a valid irq_flags value,
which forces us to handle the parameters differently.
This patch introduces a new flag MPU401_INFO_IRQ_HOOK for when the
device interrupt is handled by the caller, and makes the allocation of
the interrupt to depend only on the irq parameter. As suggested by
Takashi, the irq_flags parameter was dropped because, when used, it had
the constant value IRQF_DISABLED.
Signed-off-by: Clemens Ladisch <clemens@ladisch.de>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2011-09-13 09:24:41 +00:00
|
|
|
c->mpu401_2_info_flags |
|
|
|
|
MPU401_INFO_INTEGRATED | MPU401_INFO_IRQ_HOOK,
|
|
|
|
-1, &ice->rmidi[1]);
|
2008-09-07 10:17:02 +00:00
|
|
|
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
2006-04-26 16:13:59 +00:00
|
|
|
if (c->mpu401_2_name)
|
2011-03-31 01:57:33 +00:00
|
|
|
/* Preferred name available in card_info */
|
2006-04-26 16:13:59 +00:00
|
|
|
snprintf(ice->rmidi[1]->name,
|
|
|
|
sizeof(ice->rmidi[1]->name),
|
|
|
|
"%s %d", c->mpu401_2_name,
|
|
|
|
card->number);
|
|
|
|
}
|
2005-04-16 22:20:36 +00:00
|
|
|
}
|
|
|
|
|
2006-02-02 06:56:54 +00:00
|
|
|
snd_ice1712_set_input_clock_source(ice, 0);
|
|
|
|
|
2005-04-16 22:20:36 +00:00
|
|
|
sprintf(card->longname, "%s at 0x%lx, irq %i",
|
|
|
|
card->shortname, ice->port, ice->irq);
|
|
|
|
|
2008-09-07 10:17:02 +00:00
|
|
|
err = snd_card_register(card);
|
2021-07-15 07:58:57 +00:00
|
|
|
if (err < 0)
|
2005-04-16 22:20:36 +00:00
|
|
|
return err;
|
|
|
|
pci_set_drvdata(pci, card);
|
|
|
|
dev++;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-03-30 21:37:30 +00:00
|
|
|
#ifdef CONFIG_PM_SLEEP
|
|
|
|
static int snd_ice1712_suspend(struct device *dev)
|
|
|
|
{
|
|
|
|
struct snd_card *card = dev_get_drvdata(dev);
|
|
|
|
struct snd_ice1712 *ice = card->private_data;
|
|
|
|
|
|
|
|
if (!ice->pm_suspend_enabled)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
snd_power_change_state(card, SNDRV_CTL_POWER_D3hot);
|
|
|
|
|
|
|
|
snd_ac97_suspend(ice->ac97);
|
|
|
|
|
2014-04-03 21:09:38 +00:00
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
ice->pm_saved_is_spdif_master = is_spdif_master(ice);
|
|
|
|
ice->pm_saved_spdif_ctrl = inw(ICEMT(ice, ROUTE_SPDOUT));
|
|
|
|
ice->pm_saved_route = inw(ICEMT(ice, ROUTE_PSDOUT03));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
|
2014-03-30 21:37:30 +00:00
|
|
|
if (ice->pm_suspend)
|
|
|
|
ice->pm_suspend(ice);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int snd_ice1712_resume(struct device *dev)
|
|
|
|
{
|
|
|
|
struct snd_card *card = dev_get_drvdata(dev);
|
|
|
|
struct snd_ice1712 *ice = card->private_data;
|
2014-04-03 21:09:38 +00:00
|
|
|
int rate;
|
2014-03-30 21:37:30 +00:00
|
|
|
|
|
|
|
if (!ice->pm_suspend_enabled)
|
|
|
|
return 0;
|
|
|
|
|
2014-04-03 21:09:38 +00:00
|
|
|
if (ice->cur_rate)
|
|
|
|
rate = ice->cur_rate;
|
|
|
|
else
|
|
|
|
rate = PRO_RATE_DEFAULT;
|
|
|
|
|
2014-03-30 21:37:30 +00:00
|
|
|
if (snd_ice1712_chip_init(ice) < 0) {
|
|
|
|
snd_card_disconnect(card);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
2014-04-03 21:09:38 +00:00
|
|
|
ice->cur_rate = rate;
|
|
|
|
|
2014-03-30 21:37:30 +00:00
|
|
|
if (ice->pm_resume)
|
|
|
|
ice->pm_resume(ice);
|
|
|
|
|
2014-04-03 21:09:38 +00:00
|
|
|
if (ice->pm_saved_is_spdif_master) {
|
|
|
|
/* switching to external clock via SPDIF */
|
|
|
|
spin_lock_irq(&ice->reg_lock);
|
|
|
|
outb(inb(ICEMT(ice, RATE)) | ICE1712_SPDIF_MASTER,
|
|
|
|
ICEMT(ice, RATE));
|
|
|
|
spin_unlock_irq(&ice->reg_lock);
|
|
|
|
snd_ice1712_set_input_clock_source(ice, 1);
|
|
|
|
} else {
|
|
|
|
/* internal on-card clock */
|
|
|
|
snd_ice1712_set_pro_rate(ice, rate, 1);
|
|
|
|
snd_ice1712_set_input_clock_source(ice, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
outw(ice->pm_saved_spdif_ctrl, ICEMT(ice, ROUTE_SPDOUT));
|
|
|
|
outw(ice->pm_saved_route, ICEMT(ice, ROUTE_PSDOUT03));
|
|
|
|
|
2014-11-17 10:28:02 +00:00
|
|
|
snd_ac97_resume(ice->ac97);
|
2014-03-30 21:37:30 +00:00
|
|
|
|
|
|
|
snd_power_change_state(card, SNDRV_CTL_POWER_D0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static SIMPLE_DEV_PM_OPS(snd_ice1712_pm, snd_ice1712_suspend, snd_ice1712_resume);
|
|
|
|
#define SND_VT1712_PM_OPS &snd_ice1712_pm
|
|
|
|
#else
|
|
|
|
#define SND_VT1712_PM_OPS NULL
|
|
|
|
#endif /* CONFIG_PM_SLEEP */
|
|
|
|
|
2012-04-24 10:25:00 +00:00
|
|
|
static struct pci_driver ice1712_driver = {
|
2011-06-10 14:20:20 +00:00
|
|
|
.name = KBUILD_MODNAME,
|
2005-04-16 22:20:36 +00:00
|
|
|
.id_table = snd_ice1712_ids,
|
|
|
|
.probe = snd_ice1712_probe,
|
2014-03-30 21:37:30 +00:00
|
|
|
.driver = {
|
|
|
|
.pm = SND_VT1712_PM_OPS,
|
|
|
|
},
|
2005-04-16 22:20:36 +00:00
|
|
|
};
|
|
|
|
|
2012-04-24 10:25:00 +00:00
|
|
|
module_pci_driver(ice1712_driver);
|