Commit ab93c7ae authored by Krzysztof Helt's avatar Krzysztof Helt Committed by Jaroslav Kysela

[ALSA] sparc dbri: hardware constrains added

This patch adds ALSA hardware constrains so stereo is possible
only with 16-bit format. It contains small cleanups to ring
buffered code as well.
Signed-off-by: default avatarKrzysztof Helt <krzysztof.h1@wp.pl>
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
Signed-off-by: default avatarJaroslav Kysela <perex@suse.cz>
parent 1be54c82
...@@ -85,7 +85,7 @@ MODULE_PARM_DESC(id, "ID string for Sun DBRI soundcard."); ...@@ -85,7 +85,7 @@ MODULE_PARM_DESC(id, "ID string for Sun DBRI soundcard.");
module_param_array(enable, bool, NULL, 0444); module_param_array(enable, bool, NULL, 0444);
MODULE_PARM_DESC(enable, "Enable Sun DBRI soundcard."); MODULE_PARM_DESC(enable, "Enable Sun DBRI soundcard.");
#define DBRI_DEBUG #undef DBRI_DEBUG
#define D_INT (1<<0) #define D_INT (1<<0)
#define D_GEN (1<<1) #define D_GEN (1<<1)
...@@ -160,7 +160,7 @@ static struct { ...@@ -160,7 +160,7 @@ static struct {
/* { NA, (1 << 4), (5 << 3) }, */ /* { NA, (1 << 4), (5 << 3) }, */
{ 48000, (1 << 4), (6 << 3) }, { 48000, (1 << 4), (6 << 3) },
{ 9600, (1 << 4), (7 << 3) }, { 9600, (1 << 4), (7 << 3) },
{ 5513, (2 << 4), (0 << 3) }, /* Actually 5512.5 */ { 5512, (2 << 4), (0 << 3) }, /* Actually 5512.5 */
{ 11025, (2 << 4), (1 << 3) }, { 11025, (2 << 4), (1 << 3) },
{ 18900, (2 << 4), (2 << 3) }, { 18900, (2 << 4), (2 << 3) },
{ 22050, (2 << 4), (3 << 3) }, { 22050, (2 << 4), (3 << 3) },
...@@ -628,8 +628,6 @@ to send them to the DBRI. ...@@ -628,8 +628,6 @@ to send them to the DBRI.
*/ */
static void dbri_process_interrupt_buffer(struct snd_dbri * dbri);
#define MAXLOOPS 10 #define MAXLOOPS 10
/* /*
* Wait for the current command string to execute * Wait for the current command string to execute
...@@ -669,15 +667,15 @@ static s32 *dbri_cmdlock(struct snd_dbri * dbri, int len) ...@@ -669,15 +667,15 @@ static s32 *dbri_cmdlock(struct snd_dbri * dbri, int len)
} }
/* /*
* Send prepared cmd string. It works by writting a JMP cmd into * Send prepared cmd string. It works by writting a JUMP cmd into
* the last WAIT cmd and force DBRI to reread the cmd. * the last WAIT cmd and force DBRI to reread the cmd.
* The JMP cmd points to the new cmd string. * The JUMP cmd points to the new cmd string.
* It also releases the cmdlock spinlock. * It also releases the cmdlock spinlock.
*/ */
static void dbri_cmdsend(struct snd_dbri * dbri, s32 * cmd,int len) static void dbri_cmdsend(struct snd_dbri * dbri, s32 * cmd,int len)
{ {
s32 *ptr;
s32 tmp, addr; s32 tmp, addr;
unsigned long flags;
static int wait_id = 0; static int wait_id = 0;
wait_id++; wait_id++;
...@@ -691,14 +689,17 @@ static void dbri_cmdsend(struct snd_dbri * dbri, s32 * cmd,int len) ...@@ -691,14 +689,17 @@ static void dbri_cmdsend(struct snd_dbri * dbri, s32 * cmd,int len)
*(dbri->cmdptr) = DBRI_CMD(D_JUMP, 0, 0); *(dbri->cmdptr) = DBRI_CMD(D_JUMP, 0, 0);
#ifdef DBRI_DEBUG #ifdef DBRI_DEBUG
if (cmd > dbri->cmdptr ) if (cmd > dbri->cmdptr) {
s32 *ptr;
for (ptr = dbri->cmdptr; ptr < cmd+2; ptr++) { for (ptr = dbri->cmdptr; ptr < cmd+2; ptr++) {
dprintk(D_CMD, "cmd: %lx:%08x\n", (unsigned long)ptr, *ptr); dprintk(D_CMD, "cmd: %lx:%08x\n", (unsigned long)ptr, *ptr);
} }
else { } else {
ptr = dbri->cmdptr; s32 *ptr = dbri->cmdptr;
dprintk(D_CMD, "cmd: %lx:%08x\n", (unsigned long)ptr, *ptr); dprintk(D_CMD, "cmd: %lx:%08x\n", (unsigned long)ptr, *ptr);
ptr = dbri->cmdptr+1; ptr++;
dprintk(D_CMD, "cmd: %lx:%08x\n", (unsigned long)ptr, *ptr); dprintk(D_CMD, "cmd: %lx:%08x\n", (unsigned long)ptr, *ptr);
for (ptr = dbri->dma->cmd; ptr < cmd+2; ptr++) { for (ptr = dbri->dma->cmd; ptr < cmd+2; ptr++) {
dprintk(D_CMD, "cmd: %lx:%08x\n", (unsigned long)ptr, *ptr); dprintk(D_CMD, "cmd: %lx:%08x\n", (unsigned long)ptr, *ptr);
...@@ -706,10 +707,12 @@ static void dbri_cmdsend(struct snd_dbri * dbri, s32 * cmd,int len) ...@@ -706,10 +707,12 @@ static void dbri_cmdsend(struct snd_dbri * dbri, s32 * cmd,int len)
} }
#endif #endif
spin_lock_irqsave(&dbri->lock, flags);
/* Reread the last command */ /* Reread the last command */
tmp = sbus_readl(dbri->regs + REG0); tmp = sbus_readl(dbri->regs + REG0);
tmp |= D_P; tmp |= D_P;
sbus_writel(tmp, dbri->regs + REG0); sbus_writel(tmp, dbri->regs + REG0);
spin_unlock_irqrestore(&dbri->lock, flags);
dbri->cmdptr = cmd; dbri->cmdptr = cmd;
spin_unlock(&dbri->cmdlock); spin_unlock(&dbri->cmdlock);
...@@ -1549,8 +1552,7 @@ static int cs4215_prepare(struct snd_dbri * dbri, unsigned int rate, ...@@ -1549,8 +1552,7 @@ static int cs4215_prepare(struct snd_dbri * dbri, unsigned int rate,
CS4215_BSEL_128 | CS4215_FREQ[freq_idx].xtal; CS4215_BSEL_128 | CS4215_FREQ[freq_idx].xtal;
dbri->mm.channels = channels; dbri->mm.channels = channels;
/* Stereo bit: 8 bit stereo not working yet. */ if (channels == 2)
if ((channels > 1) && (dbri->mm.precision == 16))
dbri->mm.ctrl[1] |= CS4215_DFR_STEREO; dbri->mm.ctrl[1] |= CS4215_DFR_STEREO;
ret = cs4215_setctrl(dbri); ret = cs4215_setctrl(dbri);
...@@ -1624,7 +1626,7 @@ interrupts are disabled. ...@@ -1624,7 +1626,7 @@ interrupts are disabled.
/* xmit_descs() /* xmit_descs()
* *
* Transmit the current TD's for recording/playing, if needed. * Starts transmiting the current TD's for recording/playing.
* For playback, ALSA has filled the DMA memory with new data (we hope). * For playback, ALSA has filled the DMA memory with new data (we hope).
*/ */
static void xmit_descs(struct snd_dbri *dbri) static void xmit_descs(struct snd_dbri *dbri)
...@@ -1699,9 +1701,9 @@ static void xmit_descs(struct snd_dbri *dbri) ...@@ -1699,9 +1701,9 @@ static void xmit_descs(struct snd_dbri *dbri)
* them as available. Stops when the first descriptor is found without * them as available. Stops when the first descriptor is found without
* TBC (Transmit Buffer Complete) set, or we've run through them all. * TBC (Transmit Buffer Complete) set, or we've run through them all.
* *
* The DMA buffers are not released, but re-used. Since the transmit buffer * The DMA buffers are not released. They form a ring buffer and
* descriptors are not clobbered, they can be re-submitted as is. This is * they are filled by ALSA while others are transmitted by DMA.
* done by the xmit_descs() tasklet above since that could take longer. *
*/ */
static void transmission_complete_intr(struct snd_dbri * dbri, int pipe) static void transmission_complete_intr(struct snd_dbri * dbri, int pipe)
...@@ -1944,8 +1946,8 @@ static struct snd_pcm_hardware snd_dbri_pcm_hw = { ...@@ -1944,8 +1946,8 @@ static struct snd_pcm_hardware snd_dbri_pcm_hw = {
SNDRV_PCM_FMTBIT_A_LAW | SNDRV_PCM_FMTBIT_A_LAW |
SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_U8 |
SNDRV_PCM_FMTBIT_S16_BE, SNDRV_PCM_FMTBIT_S16_BE,
.rates = SNDRV_PCM_RATE_8000_48000, .rates = SNDRV_PCM_RATE_8000_48000 | SNDRV_PCM_RATE_5512,
.rate_min = 8000, .rate_min = 5512,
.rate_max = 48000, .rate_max = 48000,
.channels_min = 1, .channels_min = 1,
.channels_max = 2, .channels_max = 2,
...@@ -1956,6 +1958,39 @@ static struct snd_pcm_hardware snd_dbri_pcm_hw = { ...@@ -1956,6 +1958,39 @@ static struct snd_pcm_hardware snd_dbri_pcm_hw = {
.periods_max = 1024, .periods_max = 1024,
}; };
static int snd_hw_rule_format(struct snd_pcm_hw_params *params,
struct snd_pcm_hw_rule *rule)
{
struct snd_interval *c = hw_param_interval(params,
SNDRV_PCM_HW_PARAM_CHANNELS);
struct snd_mask *f = hw_param_mask(params, SNDRV_PCM_HW_PARAM_FORMAT);
struct snd_mask fmt;
snd_mask_any(&fmt);
if (c->min > 1) {
fmt.bits[0] &= SNDRV_PCM_FMTBIT_S16_BE;
return snd_mask_refine(f, &fmt);
}
return 0;
}
static int snd_hw_rule_channels(struct snd_pcm_hw_params *params,
struct snd_pcm_hw_rule *rule)
{
struct snd_interval *c = hw_param_interval(params,
SNDRV_PCM_HW_PARAM_CHANNELS);
struct snd_mask *f = hw_param_mask(params, SNDRV_PCM_HW_PARAM_FORMAT);
struct snd_interval ch;
snd_interval_any(&ch);
if (!(f->bits[0] & SNDRV_PCM_FMTBIT_S16_BE)) {
ch.min = ch.max = 1;
ch.integer = 1;
return snd_interval_refine(c, &ch);
}
return 0;
}
static int snd_dbri_open(struct snd_pcm_substream *substream) static int snd_dbri_open(struct snd_pcm_substream *substream)
{ {
struct snd_dbri *dbri = snd_pcm_substream_chip(substream); struct snd_dbri *dbri = snd_pcm_substream_chip(substream);
...@@ -1973,6 +2008,14 @@ static int snd_dbri_open(struct snd_pcm_substream *substream) ...@@ -1973,6 +2008,14 @@ static int snd_dbri_open(struct snd_pcm_substream *substream)
info->pipe = -1; info->pipe = -1;
spin_unlock_irqrestore(&dbri->lock, flags); spin_unlock_irqrestore(&dbri->lock, flags);
snd_pcm_hw_rule_add(runtime,0,SNDRV_PCM_HW_PARAM_CHANNELS,
snd_hw_rule_format, 0, SNDRV_PCM_HW_PARAM_FORMAT,
-1);
snd_pcm_hw_rule_add(runtime,0,SNDRV_PCM_HW_PARAM_FORMAT,
snd_hw_rule_channels, 0,
SNDRV_PCM_HW_PARAM_CHANNELS,
-1);
cs4215_open(dbri); cs4215_open(dbri);
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment