2005-04-16 15:20:36 -07:00
/*
* ALSA modem driver for VIA VT82xx ( South Bridge )
*
* VT82C686A / B / C , VT8233A / C , VT8235
*
* Copyright ( c ) 2000 Jaroslav Kysela < perex @ suse . cz >
* Tjeerd . Mulder < Tjeerd . Mulder @ fujitsu - siemens . com >
* 2002 Takashi Iwai < tiwai @ suse . de >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
*/
/*
* Changes :
*
2005-10-10 11:45:31 +02:00
* Sep . 2 , 2004 Sasha Khapyorsky < sashak @ alsa - project . org >
2005-04-16 15:20:36 -07:00
* Modified from original audio driver ' via82xx . c ' to support AC97
* modems .
*/
# include <sound/driver.h>
# include <asm/io.h>
# include <linux/delay.h>
# include <linux/interrupt.h>
# include <linux/init.h>
# include <linux/pci.h>
# include <linux/slab.h>
# include <linux/moduleparam.h>
# include <sound/core.h>
# include <sound/pcm.h>
# include <sound/pcm_params.h>
# include <sound/info.h>
# include <sound/ac97_codec.h>
# include <sound/initval.h>
#if 0
# define POINTER_DEBUG
# endif
MODULE_AUTHOR ( " Jaroslav Kysela <perex@suse.cz> " ) ;
MODULE_DESCRIPTION ( " VIA VT82xx modem " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_SUPPORTED_DEVICE ( " {{VIA,VT82C686A/B/C modem,pci}} " ) ;
2005-10-04 08:46:51 +02:00
static int index = - 2 ; /* Exclude the first card */
static char * id = SNDRV_DEFAULT_STR1 ; /* ID for this card */
static int ac97_clock = 48000 ;
2005-04-16 15:20:36 -07:00
2005-10-04 08:46:51 +02:00
module_param ( index , int , 0444 ) ;
2005-04-16 15:20:36 -07:00
MODULE_PARM_DESC ( index , " Index value for VIA 82xx bridge. " ) ;
2005-10-04 08:46:51 +02:00
module_param ( id , charp , 0444 ) ;
2005-04-16 15:20:36 -07:00
MODULE_PARM_DESC ( id , " ID string for VIA 82xx bridge. " ) ;
2005-10-04 08:46:51 +02:00
module_param ( ac97_clock , int , 0444 ) ;
2005-04-16 15:20:36 -07:00
MODULE_PARM_DESC ( ac97_clock , " AC'97 codec clock (default 48000Hz). " ) ;
2005-10-06 13:47:23 +02:00
/* just for backward compatibility */
static int enable ;
2005-10-20 16:53:49 +02:00
module_param ( enable , bool , 0444 ) ;
2005-10-06 13:47:23 +02:00
2005-04-16 15:20:36 -07:00
/*
* Direct registers
*/
# define VIAREG(via, x) ((via)->port + VIA_REG_##x)
# define VIADEV_REG(viadev, x) ((viadev)->port + VIA_REG_##x)
/* common offsets */
# define VIA_REG_OFFSET_STATUS 0x00 /* byte - channel status */
# define VIA_REG_STAT_ACTIVE 0x80 /* RO */
# define VIA_REG_STAT_PAUSED 0x40 /* RO */
# define VIA_REG_STAT_TRIGGER_QUEUED 0x08 /* RO */
# define VIA_REG_STAT_STOPPED 0x04 /* RWC */
# define VIA_REG_STAT_EOL 0x02 /* RWC */
# define VIA_REG_STAT_FLAG 0x01 /* RWC */
# define VIA_REG_OFFSET_CONTROL 0x01 /* byte - channel control */
# define VIA_REG_CTRL_START 0x80 /* WO */
# define VIA_REG_CTRL_TERMINATE 0x40 /* WO */
# define VIA_REG_CTRL_AUTOSTART 0x20
# define VIA_REG_CTRL_PAUSE 0x08 /* RW */
# define VIA_REG_CTRL_INT_STOP 0x04
# define VIA_REG_CTRL_INT_EOL 0x02
# define VIA_REG_CTRL_INT_FLAG 0x01
# define VIA_REG_CTRL_RESET 0x01 /* RW - probably reset? undocumented */
# define VIA_REG_CTRL_INT (VIA_REG_CTRL_INT_FLAG | VIA_REG_CTRL_INT_EOL | VIA_REG_CTRL_AUTOSTART)
# define VIA_REG_OFFSET_TYPE 0x02 /* byte - channel type (686 only) */
# define VIA_REG_TYPE_AUTOSTART 0x80 /* RW - autostart at EOL */
# define VIA_REG_TYPE_16BIT 0x20 /* RW */
# define VIA_REG_TYPE_STEREO 0x10 /* RW */
# define VIA_REG_TYPE_INT_LLINE 0x00
# define VIA_REG_TYPE_INT_LSAMPLE 0x04
# define VIA_REG_TYPE_INT_LESSONE 0x08
# define VIA_REG_TYPE_INT_MASK 0x0c
# define VIA_REG_TYPE_INT_EOL 0x02
# define VIA_REG_TYPE_INT_FLAG 0x01
# define VIA_REG_OFFSET_TABLE_PTR 0x04 /* dword - channel table pointer */
# define VIA_REG_OFFSET_CURR_PTR 0x04 /* dword - channel current pointer */
# define VIA_REG_OFFSET_STOP_IDX 0x08 /* dword - stop index, channel type, sample rate */
# define VIA_REG_OFFSET_CURR_COUNT 0x0c /* dword - channel current count (24 bit) */
# define VIA_REG_OFFSET_CURR_INDEX 0x0f /* byte - channel current index (for via8233 only) */
# define DEFINE_VIA_REGSET(name,val) \
enum { \
VIA_REG_ # # name # # _STATUS = ( val ) , \
VIA_REG_ # # name # # _CONTROL = ( val ) + 0x01 , \
VIA_REG_ # # name # # _TYPE = ( val ) + 0x02 , \
VIA_REG_ # # name # # _TABLE_PTR = ( val ) + 0x04 , \
VIA_REG_ # # name # # _CURR_PTR = ( val ) + 0x04 , \
VIA_REG_ # # name # # _STOP_IDX = ( val ) + 0x08 , \
VIA_REG_ # # name # # _CURR_COUNT = ( val ) + 0x0c , \
}
/* modem block */
DEFINE_VIA_REGSET ( MO , 0x40 ) ;
DEFINE_VIA_REGSET ( MI , 0x50 ) ;
/* AC'97 */
# define VIA_REG_AC97 0x80 /* dword */
# define VIA_REG_AC97_CODEC_ID_MASK (3<<30)
# define VIA_REG_AC97_CODEC_ID_SHIFT 30
# define VIA_REG_AC97_CODEC_ID_PRIMARY 0x00
# define VIA_REG_AC97_CODEC_ID_SECONDARY 0x01
# define VIA_REG_AC97_SECONDARY_VALID (1<<27)
# define VIA_REG_AC97_PRIMARY_VALID (1<<25)
# define VIA_REG_AC97_BUSY (1<<24)
# define VIA_REG_AC97_READ (1<<23)
# define VIA_REG_AC97_CMD_SHIFT 16
# define VIA_REG_AC97_CMD_MASK 0x7e
# define VIA_REG_AC97_DATA_SHIFT 0
# define VIA_REG_AC97_DATA_MASK 0xffff
# define VIA_REG_SGD_SHADOW 0x84 /* dword */
# define VIA_REG_SGD_STAT_PB_FLAG (1<<0)
# define VIA_REG_SGD_STAT_CP_FLAG (1<<1)
# define VIA_REG_SGD_STAT_FM_FLAG (1<<2)
# define VIA_REG_SGD_STAT_PB_EOL (1<<4)
# define VIA_REG_SGD_STAT_CP_EOL (1<<5)
# define VIA_REG_SGD_STAT_FM_EOL (1<<6)
# define VIA_REG_SGD_STAT_PB_STOP (1<<8)
# define VIA_REG_SGD_STAT_CP_STOP (1<<9)
# define VIA_REG_SGD_STAT_FM_STOP (1<<10)
# define VIA_REG_SGD_STAT_PB_ACTIVE (1<<12)
# define VIA_REG_SGD_STAT_CP_ACTIVE (1<<13)
# define VIA_REG_SGD_STAT_FM_ACTIVE (1<<14)
# define VIA_REG_SGD_STAT_MR_FLAG (1<<16)
# define VIA_REG_SGD_STAT_MW_FLAG (1<<17)
# define VIA_REG_SGD_STAT_MR_EOL (1<<20)
# define VIA_REG_SGD_STAT_MW_EOL (1<<21)
# define VIA_REG_SGD_STAT_MR_STOP (1<<24)
# define VIA_REG_SGD_STAT_MW_STOP (1<<25)
# define VIA_REG_SGD_STAT_MR_ACTIVE (1<<28)
# define VIA_REG_SGD_STAT_MW_ACTIVE (1<<29)
# define VIA_REG_GPI_STATUS 0x88
# define VIA_REG_GPI_INTR 0x8c
# define VIA_TBL_BIT_FLAG 0x40000000
# define VIA_TBL_BIT_EOL 0x80000000
/* pci space */
# define VIA_ACLINK_STAT 0x40
# define VIA_ACLINK_C11_READY 0x20
# define VIA_ACLINK_C10_READY 0x10
# define VIA_ACLINK_C01_READY 0x04 /* secondary codec ready */
# define VIA_ACLINK_LOWPOWER 0x02 /* low-power state */
# define VIA_ACLINK_C00_READY 0x01 /* primary codec ready */
# define VIA_ACLINK_CTRL 0x41
# define VIA_ACLINK_CTRL_ENABLE 0x80 /* 0: disable, 1: enable */
# define VIA_ACLINK_CTRL_RESET 0x40 /* 0: assert, 1: de-assert */
# define VIA_ACLINK_CTRL_SYNC 0x20 /* 0: release SYNC, 1: force SYNC hi */
# define VIA_ACLINK_CTRL_SDO 0x10 /* 0: release SDO, 1: force SDO hi */
# define VIA_ACLINK_CTRL_VRA 0x08 /* 0: disable VRA, 1: enable VRA */
# define VIA_ACLINK_CTRL_PCM 0x04 /* 0: disable PCM, 1: enable PCM */
# define VIA_ACLINK_CTRL_FM 0x02 /* via686 only */
# define VIA_ACLINK_CTRL_SB 0x01 /* via686 only */
# define VIA_ACLINK_CTRL_INIT (VIA_ACLINK_CTRL_ENABLE|\
VIA_ACLINK_CTRL_RESET | \
VIA_ACLINK_CTRL_PCM )
# define VIA_FUNC_ENABLE 0x42
# define VIA_FUNC_MIDI_PNP 0x80 /* FIXME: it's 0x40 in the datasheet! */
# define VIA_FUNC_MIDI_IRQMASK 0x40 /* FIXME: not documented! */
# define VIA_FUNC_RX2C_WRITE 0x20
# define VIA_FUNC_SB_FIFO_EMPTY 0x10
# define VIA_FUNC_ENABLE_GAME 0x08
# define VIA_FUNC_ENABLE_FM 0x04
# define VIA_FUNC_ENABLE_MIDI 0x02
# define VIA_FUNC_ENABLE_SB 0x01
# define VIA_PNP_CONTROL 0x43
# define VIA_MC97_CTRL 0x44
# define VIA_MC97_CTRL_ENABLE 0x80
# define VIA_MC97_CTRL_SECONDARY 0x40
# define VIA_MC97_CTRL_INIT (VIA_MC97_CTRL_ENABLE|\
VIA_MC97_CTRL_SECONDARY )
/*
* pcm stream
*/
struct snd_via_sg_table {
unsigned int offset ;
unsigned int size ;
} ;
# define VIA_TABLE_SIZE 255
2005-11-17 15:06:15 +01:00
struct viadev {
2005-04-16 15:20:36 -07:00
unsigned int reg_offset ;
unsigned long port ;
int direction ; /* playback = 0, capture = 1 */
2005-11-17 15:06:15 +01:00
struct snd_pcm_substream * substream ;
2005-04-16 15:20:36 -07:00
int running ;
unsigned int tbl_entries ; /* # descriptors */
struct snd_dma_buffer table ;
struct snd_via_sg_table * idx_table ;
/* for recovery from the unexpected pointer */
unsigned int lastpos ;
unsigned int bufsize ;
unsigned int bufsize2 ;
} ;
enum { TYPE_CARD_VIA82XX_MODEM = 1 } ;
# define VIA_MAX_MODEM_DEVS 2
2005-11-17 15:06:15 +01:00
struct via82xx_modem {
2005-04-16 15:20:36 -07:00
int irq ;
unsigned long port ;
unsigned int intr_mask ; /* SGD_SHADOW mask to check interrupts */
struct pci_dev * pci ;
2005-11-17 15:06:15 +01:00
struct snd_card * card ;
2005-04-16 15:20:36 -07:00
unsigned int num_devs ;
unsigned int playback_devno , capture_devno ;
2005-11-17 15:06:15 +01:00
struct viadev devs [ VIA_MAX_MODEM_DEVS ] ;
2005-04-16 15:20:36 -07:00
2005-11-17 15:06:15 +01:00
struct snd_pcm * pcms [ 2 ] ;
2005-04-16 15:20:36 -07:00
2005-11-17 15:06:15 +01:00
struct snd_ac97_bus * ac97_bus ;
struct snd_ac97 * ac97 ;
2005-04-16 15:20:36 -07:00
unsigned int ac97_clock ;
unsigned int ac97_secondary ; /* secondary AC'97 codec is present */
spinlock_t reg_lock ;
2005-11-17 15:06:15 +01:00
struct snd_info_entry * proc_entry ;
2005-04-16 15:20:36 -07:00
} ;
static struct pci_device_id snd_via82xx_modem_ids [ ] = {
{ 0x1106 , 0x3068 , PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , TYPE_CARD_VIA82XX_MODEM , } ,
{ 0 , }
} ;
MODULE_DEVICE_TABLE ( pci , snd_via82xx_modem_ids ) ;
/*
*/
/*
* allocate and initialize the descriptor buffers
* periods = number of periods
* fragsize = period size in bytes
*/
2005-11-17 15:06:15 +01:00
static int build_via_table ( struct viadev * dev , struct snd_pcm_substream * substream ,
2005-04-16 15:20:36 -07:00
struct pci_dev * pci ,
unsigned int periods , unsigned int fragsize )
{
unsigned int i , idx , ofs , rest ;
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = snd_pcm_substream_chip ( substream ) ;
2005-04-16 15:20:36 -07:00
struct snd_sg_buf * sgbuf = snd_pcm_substream_sgbuf ( substream ) ;
if ( dev - > table . area = = NULL ) {
/* the start of each lists must be aligned to 8 bytes,
* but the kernel pages are much bigger , so we don ' t care
*/
if ( snd_dma_alloc_pages ( SNDRV_DMA_TYPE_DEV , snd_dma_pci_data ( chip - > pci ) ,
PAGE_ALIGN ( VIA_TABLE_SIZE * 2 * 8 ) ,
& dev - > table ) < 0 )
return - ENOMEM ;
}
if ( ! dev - > idx_table ) {
dev - > idx_table = kmalloc ( sizeof ( * dev - > idx_table ) * VIA_TABLE_SIZE , GFP_KERNEL ) ;
if ( ! dev - > idx_table )
return - ENOMEM ;
}
/* fill the entries */
idx = 0 ;
ofs = 0 ;
for ( i = 0 ; i < periods ; i + + ) {
rest = fragsize ;
/* fill descriptors for a period.
* a period can be split to several descriptors if it ' s
* over page boundary .
*/
do {
unsigned int r ;
unsigned int flag ;
if ( idx > = VIA_TABLE_SIZE ) {
snd_printk ( KERN_ERR " via82xx: too much table size! \n " ) ;
return - EINVAL ;
}
( ( u32 * ) dev - > table . area ) [ idx < < 1 ] = cpu_to_le32 ( ( u32 ) snd_pcm_sgbuf_get_addr ( sgbuf , ofs ) ) ;
r = PAGE_SIZE - ( ofs % PAGE_SIZE ) ;
if ( rest < r )
r = rest ;
rest - = r ;
if ( ! rest ) {
if ( i = = periods - 1 )
flag = VIA_TBL_BIT_EOL ; /* buffer boundary */
else
flag = VIA_TBL_BIT_FLAG ; /* period boundary */
} else
flag = 0 ; /* period continues to the next */
// printk("via: tbl %d: at %d size %d (rest %d)\n", idx, ofs, r, rest);
( ( u32 * ) dev - > table . area ) [ ( idx < < 1 ) + 1 ] = cpu_to_le32 ( r | flag ) ;
dev - > idx_table [ idx ] . offset = ofs ;
dev - > idx_table [ idx ] . size = r ;
ofs + = r ;
idx + + ;
} while ( rest > 0 ) ;
}
dev - > tbl_entries = idx ;
dev - > bufsize = periods * fragsize ;
dev - > bufsize2 = dev - > bufsize / 2 ;
return 0 ;
}
2005-11-17 15:06:15 +01:00
static int clean_via_table ( struct viadev * dev , struct snd_pcm_substream * substream ,
2005-04-16 15:20:36 -07:00
struct pci_dev * pci )
{
if ( dev - > table . area ) {
snd_dma_free_pages ( & dev - > table ) ;
dev - > table . area = NULL ;
}
[ALSA] Remove redundant NULL checks before kfree
Timer Midlevel,ALSA sequencer,ALSA<-OSS sequencer,Digigram VX core
I2C tea6330t,GUS Library,VIA82xx driver,VIA82xx-modem driver
CA0106 driver,CS46xx driver,EMU10K1/EMU10K2 driver,YMFPCI driver
Digigram VX Pocket driver,Common EMU synth,USB generic driver,USB USX2Y
Checking a pointer for NULL before calling kfree() on it is redundant,
kfree() deals with NULL pointers just fine.
This patch removes such checks from sound/
This patch also makes another, but closely related, change.
It avoids casting pointers about to be kfree()'ed.
Signed-off-by: Jesper Juhl <juhl-lkml@dif.dk>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2005-05-30 17:30:32 +02:00
kfree ( dev - > idx_table ) ;
dev - > idx_table = NULL ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
/*
* Basic I / O
*/
2005-11-17 15:06:15 +01:00
static inline unsigned int snd_via82xx_codec_xread ( struct via82xx_modem * chip )
2005-04-16 15:20:36 -07:00
{
return inl ( VIAREG ( chip , AC97 ) ) ;
}
2005-11-17 15:06:15 +01:00
static inline void snd_via82xx_codec_xwrite ( struct via82xx_modem * chip , unsigned int val )
2005-04-16 15:20:36 -07:00
{
outl ( val , VIAREG ( chip , AC97 ) ) ;
}
2005-11-17 15:06:15 +01:00
static int snd_via82xx_codec_ready ( struct via82xx_modem * chip , int secondary )
2005-04-16 15:20:36 -07:00
{
unsigned int timeout = 1000 ; /* 1ms */
unsigned int val ;
while ( timeout - - > 0 ) {
udelay ( 1 ) ;
if ( ! ( ( val = snd_via82xx_codec_xread ( chip ) ) & VIA_REG_AC97_BUSY ) )
return val & 0xffff ;
}
2005-11-17 15:06:15 +01:00
snd_printk ( KERN_ERR " codec_ready: codec %i is not ready [0x%x] \n " ,
secondary , snd_via82xx_codec_xread ( chip ) ) ;
2005-04-16 15:20:36 -07:00
return - EIO ;
}
2005-11-17 15:06:15 +01:00
static int snd_via82xx_codec_valid ( struct via82xx_modem * chip , int secondary )
2005-04-16 15:20:36 -07:00
{
unsigned int timeout = 1000 ; /* 1ms */
unsigned int val , val1 ;
unsigned int stat = ! secondary ? VIA_REG_AC97_PRIMARY_VALID :
VIA_REG_AC97_SECONDARY_VALID ;
while ( timeout - - > 0 ) {
val = snd_via82xx_codec_xread ( chip ) ;
val1 = val & ( VIA_REG_AC97_BUSY | stat ) ;
if ( val1 = = stat )
return val & 0xffff ;
udelay ( 1 ) ;
}
return - EIO ;
}
2005-11-17 15:06:15 +01:00
static void snd_via82xx_codec_wait ( struct snd_ac97 * ac97 )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = ac97 - > private_data ;
2005-04-16 15:20:36 -07:00
int err ;
err = snd_via82xx_codec_ready ( chip , ac97 - > num ) ;
/* here we need to wait fairly for long time.. */
[ALSA] sound/pci: fix-up sleeping paths
ENS1370/1+ driver,ES1968 driver,Intel8x0 driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,ALI5451 driver,CS46xx driver
MIXART driver,RME HDSP driver,Trident driver,YMFPCI driver
Description: Fix-up sleeping in sound/pci. These changes fall under the
following two categories:
1) Replace schedule_timeout() with msleep() to guarantee the
task delays as expected. This also involved replacing/removing
custom sleep functions.
2) Do not assume jiffies will only increment by one if you
request a 1 jiffy sleep, i.e. use time_after/time_before in
while loops.
Signed-off-by: Nishanth Aravamudan <nacc@us.ibm.com>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2005-07-09 10:13:22 +02:00
msleep ( 500 ) ;
2005-04-16 15:20:36 -07:00
}
2005-11-17 15:06:15 +01:00
static void snd_via82xx_codec_write ( struct snd_ac97 * ac97 ,
2005-04-16 15:20:36 -07:00
unsigned short reg ,
unsigned short val )
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = ac97 - > private_data ;
2005-04-16 15:20:36 -07:00
unsigned int xval ;
2005-05-29 15:10:07 +02:00
if ( reg = = AC97_GPIO_STATUS ) {
outl ( val , VIAREG ( chip , GPI_STATUS ) ) ;
return ;
}
2005-04-16 15:20:36 -07:00
xval = ! ac97 - > num ? VIA_REG_AC97_CODEC_ID_PRIMARY : VIA_REG_AC97_CODEC_ID_SECONDARY ;
xval < < = VIA_REG_AC97_CODEC_ID_SHIFT ;
xval | = reg < < VIA_REG_AC97_CMD_SHIFT ;
xval | = val < < VIA_REG_AC97_DATA_SHIFT ;
snd_via82xx_codec_xwrite ( chip , xval ) ;
snd_via82xx_codec_ready ( chip , ac97 - > num ) ;
}
2005-11-17 15:06:15 +01:00
static unsigned short snd_via82xx_codec_read ( struct snd_ac97 * ac97 , unsigned short reg )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = ac97 - > private_data ;
2005-04-16 15:20:36 -07:00
unsigned int xval , val = 0xffff ;
int again = 0 ;
xval = ac97 - > num < < VIA_REG_AC97_CODEC_ID_SHIFT ;
xval | = ac97 - > num ? VIA_REG_AC97_SECONDARY_VALID : VIA_REG_AC97_PRIMARY_VALID ;
xval | = VIA_REG_AC97_READ ;
xval | = ( reg & 0x7f ) < < VIA_REG_AC97_CMD_SHIFT ;
while ( 1 ) {
if ( again + + > 3 ) {
2005-11-17 15:06:15 +01:00
snd_printk ( KERN_ERR " codec_read: codec %i is not valid [0x%x] \n " ,
ac97 - > num , snd_via82xx_codec_xread ( chip ) ) ;
2005-04-16 15:20:36 -07:00
return 0xffff ;
}
snd_via82xx_codec_xwrite ( chip , xval ) ;
udelay ( 20 ) ;
if ( snd_via82xx_codec_valid ( chip , ac97 - > num ) > = 0 ) {
udelay ( 25 ) ;
val = snd_via82xx_codec_xread ( chip ) ;
break ;
}
}
return val & 0xffff ;
}
2005-11-17 15:06:15 +01:00
static void snd_via82xx_channel_reset ( struct via82xx_modem * chip , struct viadev * viadev )
2005-04-16 15:20:36 -07:00
{
outb ( VIA_REG_CTRL_PAUSE | VIA_REG_CTRL_TERMINATE | VIA_REG_CTRL_RESET ,
VIADEV_REG ( viadev , OFFSET_CONTROL ) ) ;
inb ( VIADEV_REG ( viadev , OFFSET_CONTROL ) ) ;
udelay ( 50 ) ;
/* disable interrupts */
outb ( 0x00 , VIADEV_REG ( viadev , OFFSET_CONTROL ) ) ;
/* clear interrupts */
outb ( 0x03 , VIADEV_REG ( viadev , OFFSET_STATUS ) ) ;
outb ( 0x00 , VIADEV_REG ( viadev , OFFSET_TYPE ) ) ; /* for via686 */
// outl(0, VIADEV_REG(viadev, OFFSET_CURR_PTR));
viadev - > lastpos = 0 ;
}
/*
* Interrupt handler
*/
static irqreturn_t snd_via82xx_interrupt ( int irq , void * dev_id , struct pt_regs * regs )
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = dev_id ;
2005-04-16 15:20:36 -07:00
unsigned int status ;
unsigned int i ;
status = inl ( VIAREG ( chip , SGD_SHADOW ) ) ;
if ( ! ( status & chip - > intr_mask ) ) {
return IRQ_NONE ;
}
// _skip_sgd:
/* check status for each stream */
spin_lock ( & chip - > reg_lock ) ;
for ( i = 0 ; i < chip - > num_devs ; i + + ) {
2005-11-17 15:06:15 +01:00
struct viadev * viadev = & chip - > devs [ i ] ;
2005-04-16 15:20:36 -07:00
unsigned char c_status = inb ( VIADEV_REG ( viadev , OFFSET_STATUS ) ) ;
c_status & = ( VIA_REG_STAT_EOL | VIA_REG_STAT_FLAG | VIA_REG_STAT_STOPPED ) ;
if ( ! c_status )
continue ;
if ( viadev - > substream & & viadev - > running ) {
spin_unlock ( & chip - > reg_lock ) ;
snd_pcm_period_elapsed ( viadev - > substream ) ;
spin_lock ( & chip - > reg_lock ) ;
}
outb ( c_status , VIADEV_REG ( viadev , OFFSET_STATUS ) ) ; /* ack */
}
spin_unlock ( & chip - > reg_lock ) ;
return IRQ_HANDLED ;
}
/*
* PCM callbacks
*/
/*
* trigger callback
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_pcm_trigger ( struct snd_pcm_substream * substream , int cmd )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = snd_pcm_substream_chip ( substream ) ;
struct viadev * viadev = substream - > runtime - > private_data ;
2005-04-16 15:20:36 -07:00
unsigned char val = 0 ;
switch ( cmd ) {
case SNDRV_PCM_TRIGGER_START :
2005-08-18 13:43:12 +02:00
case SNDRV_PCM_TRIGGER_SUSPEND :
2005-04-16 15:20:36 -07:00
val | = VIA_REG_CTRL_START ;
viadev - > running = 1 ;
break ;
case SNDRV_PCM_TRIGGER_STOP :
val = VIA_REG_CTRL_TERMINATE ;
viadev - > running = 0 ;
break ;
case SNDRV_PCM_TRIGGER_PAUSE_PUSH :
val | = VIA_REG_CTRL_PAUSE ;
viadev - > running = 0 ;
break ;
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE :
viadev - > running = 1 ;
break ;
default :
return - EINVAL ;
}
outb ( val , VIADEV_REG ( viadev , OFFSET_CONTROL ) ) ;
if ( cmd = = SNDRV_PCM_TRIGGER_STOP )
snd_via82xx_channel_reset ( chip , viadev ) ;
return 0 ;
}
/*
* pointer callbacks
*/
/*
* calculate the linear position at the given sg - buffer index and the rest count
*/
# define check_invalid_pos(viadev,pos) \
2005-11-17 15:06:15 +01:00
( ( pos ) < viadev - > lastpos & & ( ( pos ) > = viadev - > bufsize2 | | \
viadev - > lastpos < viadev - > bufsize2 ) )
2005-04-16 15:20:36 -07:00
2005-11-17 15:06:15 +01:00
static inline unsigned int calc_linear_pos ( struct viadev * viadev , unsigned int idx ,
unsigned int count )
2005-04-16 15:20:36 -07:00
{
unsigned int size , res ;
size = viadev - > idx_table [ idx ] . size ;
res = viadev - > idx_table [ idx ] . offset + size - count ;
/* check the validity of the calculated position */
if ( size < count ) {
2005-11-17 15:06:15 +01:00
snd_printd ( KERN_ERR " invalid via82xx_cur_ptr (size = %d, count = %d) \n " ,
( int ) size , ( int ) count ) ;
2005-04-16 15:20:36 -07:00
res = viadev - > lastpos ;
} else if ( check_invalid_pos ( viadev , res ) ) {
# ifdef POINTER_DEBUG
2005-11-17 15:06:15 +01:00
printk ( KERN_DEBUG " fail: idx = %i/%i, lastpos = 0x%x, "
" bufsize2 = 0x%x, offsize = 0x%x, size = 0x%x, "
" count = 0x%x \n " , idx , viadev - > tbl_entries , viadev - > lastpos ,
viadev - > bufsize2 , viadev - > idx_table [ idx ] . offset ,
viadev - > idx_table [ idx ] . size , count ) ;
2005-04-16 15:20:36 -07:00
# endif
if ( count & & size < count ) {
2005-11-17 15:06:15 +01:00
snd_printd ( KERN_ERR " invalid via82xx_cur_ptr, "
" using last valid pointer \n " ) ;
2005-04-16 15:20:36 -07:00
res = viadev - > lastpos ;
} else {
if ( ! count )
/* bogus count 0 on the DMA boundary? */
res = viadev - > idx_table [ idx ] . offset ;
else
2005-11-17 15:06:15 +01:00
/* count register returns full size
* when end of buffer is reached
*/
2005-04-16 15:20:36 -07:00
res = viadev - > idx_table [ idx ] . offset + size ;
if ( check_invalid_pos ( viadev , res ) ) {
2005-11-17 15:06:15 +01:00
snd_printd ( KERN_ERR " invalid via82xx_cur_ptr (2), "
" using last valid pointer \n " ) ;
2005-04-16 15:20:36 -07:00
res = viadev - > lastpos ;
}
}
}
viadev - > lastpos = res ; /* remember the last position */
if ( res > = viadev - > bufsize )
res - = viadev - > bufsize ;
return res ;
}
/*
* get the current pointer on via686
*/
2005-11-17 15:06:15 +01:00
static snd_pcm_uframes_t snd_via686_pcm_pointer ( struct snd_pcm_substream * substream )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = snd_pcm_substream_chip ( substream ) ;
struct viadev * viadev = substream - > runtime - > private_data ;
2005-04-16 15:20:36 -07:00
unsigned int idx , ptr , count , res ;
snd_assert ( viadev - > tbl_entries , return 0 ) ;
if ( ! ( inb ( VIADEV_REG ( viadev , OFFSET_STATUS ) ) & VIA_REG_STAT_ACTIVE ) )
return 0 ;
spin_lock ( & chip - > reg_lock ) ;
count = inl ( VIADEV_REG ( viadev , OFFSET_CURR_COUNT ) ) & 0xffffff ;
/* The via686a does not have the current index register,
* so we need to calculate the index from CURR_PTR .
*/
ptr = inl ( VIADEV_REG ( viadev , OFFSET_CURR_PTR ) ) ;
if ( ptr < = ( unsigned int ) viadev - > table . addr )
idx = 0 ;
else /* CURR_PTR holds the address + 8 */
2005-11-17 15:06:15 +01:00
idx = ( ( ptr - ( unsigned int ) viadev - > table . addr ) / 8 - 1 ) %
viadev - > tbl_entries ;
2005-04-16 15:20:36 -07:00
res = calc_linear_pos ( viadev , idx , count ) ;
spin_unlock ( & chip - > reg_lock ) ;
return bytes_to_frames ( substream - > runtime , res ) ;
}
/*
* hw_params callback :
* allocate the buffer and build up the buffer description table
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_hw_params ( struct snd_pcm_substream * substream ,
struct snd_pcm_hw_params * hw_params )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = snd_pcm_substream_chip ( substream ) ;
struct viadev * viadev = substream - > runtime - > private_data ;
2005-04-16 15:20:36 -07:00
int err ;
err = snd_pcm_lib_malloc_pages ( substream , params_buffer_bytes ( hw_params ) ) ;
if ( err < 0 )
return err ;
err = build_via_table ( viadev , substream , chip - > pci ,
params_periods ( hw_params ) ,
params_period_bytes ( hw_params ) ) ;
if ( err < 0 )
return err ;
snd_ac97_write ( chip - > ac97 , AC97_LINE1_RATE , params_rate ( hw_params ) ) ;
snd_ac97_write ( chip - > ac97 , AC97_LINE1_LEVEL , 0 ) ;
return 0 ;
}
/*
* hw_free callback :
* clean up the buffer description table and release the buffer
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_hw_free ( struct snd_pcm_substream * substream )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = snd_pcm_substream_chip ( substream ) ;
struct viadev * viadev = substream - > runtime - > private_data ;
2005-04-16 15:20:36 -07:00
clean_via_table ( viadev , substream , chip - > pci ) ;
snd_pcm_lib_free_pages ( substream ) ;
return 0 ;
}
/*
* set up the table pointer
*/
2005-11-17 15:06:15 +01:00
static void snd_via82xx_set_table_ptr ( struct via82xx_modem * chip , struct viadev * viadev )
2005-04-16 15:20:36 -07:00
{
snd_via82xx_codec_ready ( chip , chip - > ac97_secondary ) ;
outl ( ( u32 ) viadev - > table . addr , VIADEV_REG ( viadev , OFFSET_TABLE_PTR ) ) ;
udelay ( 20 ) ;
snd_via82xx_codec_ready ( chip , chip - > ac97_secondary ) ;
}
/*
* prepare callback for playback and capture
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_pcm_prepare ( struct snd_pcm_substream * substream )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = snd_pcm_substream_chip ( substream ) ;
struct viadev * viadev = substream - > runtime - > private_data ;
2005-04-16 15:20:36 -07:00
snd_via82xx_channel_reset ( chip , viadev ) ;
/* this must be set after channel_reset */
snd_via82xx_set_table_ptr ( chip , viadev ) ;
outb ( VIA_REG_TYPE_AUTOSTART | VIA_REG_TYPE_INT_EOL | VIA_REG_TYPE_INT_FLAG ,
VIADEV_REG ( viadev , OFFSET_TYPE ) ) ;
return 0 ;
}
/*
* pcm hardware definition , identical for both playback and capture
*/
2005-11-17 15:06:15 +01:00
static struct snd_pcm_hardware snd_via82xx_hw =
2005-04-16 15:20:36 -07:00
{
. info = ( SNDRV_PCM_INFO_MMAP | SNDRV_PCM_INFO_INTERLEAVED |
SNDRV_PCM_INFO_BLOCK_TRANSFER |
SNDRV_PCM_INFO_MMAP_VALID |
2005-08-18 13:43:12 +02:00
/* SNDRV_PCM_INFO_RESUME | */
2005-04-16 15:20:36 -07:00
SNDRV_PCM_INFO_PAUSE ) ,
. formats = SNDRV_PCM_FMTBIT_U8 | SNDRV_PCM_FMTBIT_S16_LE ,
. rates = SNDRV_PCM_RATE_8000 | SNDRV_PCM_RATE_16000 | SNDRV_PCM_RATE_KNOT ,
. rate_min = 8000 ,
. rate_max = 16000 ,
. channels_min = 1 ,
. channels_max = 1 ,
. buffer_bytes_max = 128 * 1024 ,
. period_bytes_min = 32 ,
. period_bytes_max = 128 * 1024 ,
. periods_min = 2 ,
. periods_max = VIA_TABLE_SIZE / 2 ,
. fifo_size = 0 ,
} ;
/*
* open callback skeleton
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_modem_pcm_open ( struct via82xx_modem * chip , struct viadev * viadev ,
struct snd_pcm_substream * substream )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct snd_pcm_runtime * runtime = substream - > runtime ;
2005-04-16 15:20:36 -07:00
int err ;
static unsigned int rates [ ] = { 8000 , 9600 , 12000 , 16000 } ;
2005-11-17 15:06:15 +01:00
static struct snd_pcm_hw_constraint_list hw_constraints_rates = {
2005-04-16 15:20:36 -07:00
. count = ARRAY_SIZE ( rates ) ,
. list = rates ,
. mask = 0 ,
} ;
runtime - > hw = snd_via82xx_hw ;
2005-11-17 15:06:15 +01:00
if ( ( err = snd_pcm_hw_constraint_list ( runtime , 0 , SNDRV_PCM_HW_PARAM_RATE ,
& hw_constraints_rates ) ) < 0 )
2005-04-16 15:20:36 -07:00
return err ;
/* we may remove following constaint when we modify table entries
in interrupt */
if ( ( err = snd_pcm_hw_constraint_integer ( runtime , SNDRV_PCM_HW_PARAM_PERIODS ) ) < 0 )
return err ;
runtime - > private_data = viadev ;
viadev - > substream = substream ;
return 0 ;
}
/*
* open callback for playback
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_playback_open ( struct snd_pcm_substream * substream )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = snd_pcm_substream_chip ( substream ) ;
struct viadev * viadev = & chip - > devs [ chip - > playback_devno + substream - > number ] ;
2005-04-16 15:20:36 -07:00
return snd_via82xx_modem_pcm_open ( chip , viadev , substream ) ;
}
/*
* open callback for capture
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_capture_open ( struct snd_pcm_substream * substream )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = snd_pcm_substream_chip ( substream ) ;
struct viadev * viadev = & chip - > devs [ chip - > capture_devno + substream - > pcm - > device ] ;
2005-04-16 15:20:36 -07:00
return snd_via82xx_modem_pcm_open ( chip , viadev , substream ) ;
}
/*
* close callback
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_pcm_close ( struct snd_pcm_substream * substream )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct viadev * viadev = substream - > runtime - > private_data ;
2005-04-16 15:20:36 -07:00
viadev - > substream = NULL ;
return 0 ;
}
/* via686 playback callbacks */
2005-11-17 15:06:15 +01:00
static struct snd_pcm_ops snd_via686_playback_ops = {
2005-04-16 15:20:36 -07:00
. open = snd_via82xx_playback_open ,
. close = snd_via82xx_pcm_close ,
. ioctl = snd_pcm_lib_ioctl ,
. hw_params = snd_via82xx_hw_params ,
. hw_free = snd_via82xx_hw_free ,
. prepare = snd_via82xx_pcm_prepare ,
2005-05-29 15:10:07 +02:00
. trigger = snd_via82xx_pcm_trigger ,
2005-04-16 15:20:36 -07:00
. pointer = snd_via686_pcm_pointer ,
. page = snd_pcm_sgbuf_ops_page ,
} ;
/* via686 capture callbacks */
2005-11-17 15:06:15 +01:00
static struct snd_pcm_ops snd_via686_capture_ops = {
2005-04-16 15:20:36 -07:00
. open = snd_via82xx_capture_open ,
. close = snd_via82xx_pcm_close ,
. ioctl = snd_pcm_lib_ioctl ,
. hw_params = snd_via82xx_hw_params ,
. hw_free = snd_via82xx_hw_free ,
. prepare = snd_via82xx_pcm_prepare ,
2005-05-29 15:10:07 +02:00
. trigger = snd_via82xx_pcm_trigger ,
2005-04-16 15:20:36 -07:00
. pointer = snd_via686_pcm_pointer ,
. page = snd_pcm_sgbuf_ops_page ,
} ;
2005-11-17 15:06:15 +01:00
static void init_viadev ( struct via82xx_modem * chip , int idx , unsigned int reg_offset ,
int direction )
2005-04-16 15:20:36 -07:00
{
chip - > devs [ idx ] . reg_offset = reg_offset ;
chip - > devs [ idx ] . direction = direction ;
chip - > devs [ idx ] . port = chip - > port + reg_offset ;
}
/*
* create a pcm instance for via686a / b
*/
2005-11-17 15:06:15 +01:00
static int __devinit snd_via686_pcm_new ( struct via82xx_modem * chip )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct snd_pcm * pcm ;
2005-04-16 15:20:36 -07:00
int err ;
chip - > playback_devno = 0 ;
chip - > capture_devno = 1 ;
chip - > num_devs = 2 ;
chip - > intr_mask = 0x330000 ; /* FLAGS | EOL for MR, MW */
err = snd_pcm_new ( chip - > card , chip - > card - > shortname , 0 , 1 , 1 , & pcm ) ;
if ( err < 0 )
return err ;
snd_pcm_set_ops ( pcm , SNDRV_PCM_STREAM_PLAYBACK , & snd_via686_playback_ops ) ;
snd_pcm_set_ops ( pcm , SNDRV_PCM_STREAM_CAPTURE , & snd_via686_capture_ops ) ;
2005-09-29 11:48:17 +02:00
pcm - > dev_class = SNDRV_PCM_CLASS_MODEM ;
2005-04-16 15:20:36 -07:00
pcm - > private_data = chip ;
strcpy ( pcm - > name , chip - > card - > shortname ) ;
chip - > pcms [ 0 ] = pcm ;
init_viadev ( chip , 0 , VIA_REG_MO_STATUS , 0 ) ;
init_viadev ( chip , 1 , VIA_REG_MI_STATUS , 1 ) ;
if ( ( err = snd_pcm_lib_preallocate_pages_for_all ( pcm , SNDRV_DMA_TYPE_DEV_SG ,
2005-11-17 15:06:15 +01:00
snd_dma_pci_data ( chip - > pci ) ,
64 * 1024 , 128 * 1024 ) ) < 0 )
2005-04-16 15:20:36 -07:00
return err ;
return 0 ;
}
/*
* Mixer part
*/
2005-11-17 15:06:15 +01:00
static void snd_via82xx_mixer_free_ac97_bus ( struct snd_ac97_bus * bus )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = bus - > private_data ;
2005-04-16 15:20:36 -07:00
chip - > ac97_bus = NULL ;
}
2005-11-17 15:06:15 +01:00
static void snd_via82xx_mixer_free_ac97 ( struct snd_ac97 * ac97 )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = ac97 - > private_data ;
2005-04-16 15:20:36 -07:00
chip - > ac97 = NULL ;
}
2005-11-17 15:06:15 +01:00
static int __devinit snd_via82xx_mixer_new ( struct via82xx_modem * chip )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct snd_ac97_template ac97 ;
2005-04-16 15:20:36 -07:00
int err ;
2005-11-17 15:06:15 +01:00
static struct snd_ac97_bus_ops ops = {
2005-04-16 15:20:36 -07:00
. write = snd_via82xx_codec_write ,
. read = snd_via82xx_codec_read ,
. wait = snd_via82xx_codec_wait ,
} ;
if ( ( err = snd_ac97_bus ( chip - > card , 0 , & ops , chip , & chip - > ac97_bus ) ) < 0 )
return err ;
chip - > ac97_bus - > private_free = snd_via82xx_mixer_free_ac97_bus ;
chip - > ac97_bus - > clock = chip - > ac97_clock ;
memset ( & ac97 , 0 , sizeof ( ac97 ) ) ;
ac97 . private_data = chip ;
ac97 . private_free = snd_via82xx_mixer_free_ac97 ;
ac97 . pci = chip - > pci ;
ac97 . scaps = AC97_SCAP_SKIP_AUDIO ;
ac97 . num = chip - > ac97_secondary ;
if ( ( err = snd_ac97_mixer ( chip - > ac97_bus , & ac97 , & chip - > ac97 ) ) < 0 )
return err ;
return 0 ;
}
/*
* proc interface
*/
2005-11-17 15:06:15 +01:00
static void snd_via82xx_proc_read ( struct snd_info_entry * entry , struct snd_info_buffer * buffer )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = entry - > private_data ;
2005-04-16 15:20:36 -07:00
int i ;
snd_iprintf ( buffer , " %s \n \n " , chip - > card - > longname ) ;
for ( i = 0 ; i < 0xa0 ; i + = 4 ) {
snd_iprintf ( buffer , " %02x: %08x \n " , i , inl ( chip - > port + i ) ) ;
}
}
2005-11-17 15:06:15 +01:00
static void __devinit snd_via82xx_proc_init ( struct via82xx_modem * chip )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct snd_info_entry * entry ;
2005-04-16 15:20:36 -07:00
if ( ! snd_card_proc_new ( chip - > card , " via82xx " , & entry ) )
snd_info_set_text_ops ( entry , chip , 1024 , snd_via82xx_proc_read ) ;
}
/*
*
*/
2005-11-17 15:06:15 +01:00
static int snd_via82xx_chip_init ( struct via82xx_modem * chip )
2005-04-16 15:20:36 -07:00
{
unsigned int val ;
[ALSA] sound/pci: fix-up sleeping paths
ENS1370/1+ driver,ES1968 driver,Intel8x0 driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,ALI5451 driver,CS46xx driver
MIXART driver,RME HDSP driver,Trident driver,YMFPCI driver
Description: Fix-up sleeping in sound/pci. These changes fall under the
following two categories:
1) Replace schedule_timeout() with msleep() to guarantee the
task delays as expected. This also involved replacing/removing
custom sleep functions.
2) Do not assume jiffies will only increment by one if you
request a 1 jiffy sleep, i.e. use time_after/time_before in
while loops.
Signed-off-by: Nishanth Aravamudan <nacc@us.ibm.com>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2005-07-09 10:13:22 +02:00
unsigned long end_time ;
2005-04-16 15:20:36 -07:00
unsigned char pval ;
pci_read_config_byte ( chip - > pci , VIA_MC97_CTRL , & pval ) ;
if ( ( pval & VIA_MC97_CTRL_INIT ) ! = VIA_MC97_CTRL_INIT ) {
pci_write_config_byte ( chip - > pci , 0x44 , pval | VIA_MC97_CTRL_INIT ) ;
udelay ( 100 ) ;
}
pci_read_config_byte ( chip - > pci , VIA_ACLINK_STAT , & pval ) ;
if ( ! ( pval & VIA_ACLINK_C00_READY ) ) { /* codec not ready? */
/* deassert ACLink reset, force SYNC */
pci_write_config_byte ( chip - > pci , VIA_ACLINK_CTRL ,
VIA_ACLINK_CTRL_ENABLE |
VIA_ACLINK_CTRL_RESET |
VIA_ACLINK_CTRL_SYNC ) ;
udelay ( 100 ) ;
# if 1 /* FIXME: should we do full reset here for all chip models? */
pci_write_config_byte ( chip - > pci , VIA_ACLINK_CTRL , 0x00 ) ;
udelay ( 100 ) ;
# else
/* deassert ACLink reset, force SYNC (warm AC'97 reset) */
pci_write_config_byte ( chip - > pci , VIA_ACLINK_CTRL ,
VIA_ACLINK_CTRL_RESET | VIA_ACLINK_CTRL_SYNC ) ;
udelay ( 2 ) ;
# endif
/* ACLink on, deassert ACLink reset, VSR, SGD data out */
pci_write_config_byte ( chip - > pci , VIA_ACLINK_CTRL , VIA_ACLINK_CTRL_INIT ) ;
udelay ( 100 ) ;
}
pci_read_config_byte ( chip - > pci , VIA_ACLINK_CTRL , & pval ) ;
if ( ( pval & VIA_ACLINK_CTRL_INIT ) ! = VIA_ACLINK_CTRL_INIT ) {
/* ACLink on, deassert ACLink reset, VSR, SGD data out */
pci_write_config_byte ( chip - > pci , VIA_ACLINK_CTRL , VIA_ACLINK_CTRL_INIT ) ;
udelay ( 100 ) ;
}
/* wait until codec ready */
[ALSA] sound/pci: fix-up sleeping paths
ENS1370/1+ driver,ES1968 driver,Intel8x0 driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,ALI5451 driver,CS46xx driver
MIXART driver,RME HDSP driver,Trident driver,YMFPCI driver
Description: Fix-up sleeping in sound/pci. These changes fall under the
following two categories:
1) Replace schedule_timeout() with msleep() to guarantee the
task delays as expected. This also involved replacing/removing
custom sleep functions.
2) Do not assume jiffies will only increment by one if you
request a 1 jiffy sleep, i.e. use time_after/time_before in
while loops.
Signed-off-by: Nishanth Aravamudan <nacc@us.ibm.com>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2005-07-09 10:13:22 +02:00
end_time = jiffies + msecs_to_jiffies ( 750 ) ;
2005-04-16 15:20:36 -07:00
do {
pci_read_config_byte ( chip - > pci , VIA_ACLINK_STAT , & pval ) ;
if ( pval & VIA_ACLINK_C00_READY ) /* primary codec ready */
break ;
2005-10-24 15:02:37 +02:00
schedule_timeout_uninterruptible ( 1 ) ;
[ALSA] sound/pci: fix-up sleeping paths
ENS1370/1+ driver,ES1968 driver,Intel8x0 driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,ALI5451 driver,CS46xx driver
MIXART driver,RME HDSP driver,Trident driver,YMFPCI driver
Description: Fix-up sleeping in sound/pci. These changes fall under the
following two categories:
1) Replace schedule_timeout() with msleep() to guarantee the
task delays as expected. This also involved replacing/removing
custom sleep functions.
2) Do not assume jiffies will only increment by one if you
request a 1 jiffy sleep, i.e. use time_after/time_before in
while loops.
Signed-off-by: Nishanth Aravamudan <nacc@us.ibm.com>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2005-07-09 10:13:22 +02:00
} while ( time_before ( jiffies , end_time ) ) ;
2005-04-16 15:20:36 -07:00
if ( ( val = snd_via82xx_codec_xread ( chip ) ) & VIA_REG_AC97_BUSY )
2005-10-20 18:26:44 +02:00
snd_printk ( KERN_ERR " AC'97 codec is not ready [0x%x] \n " , val ) ;
2005-04-16 15:20:36 -07:00
snd_via82xx_codec_xwrite ( chip , VIA_REG_AC97_READ |
VIA_REG_AC97_SECONDARY_VALID |
( VIA_REG_AC97_CODEC_ID_SECONDARY < < VIA_REG_AC97_CODEC_ID_SHIFT ) ) ;
[ALSA] sound/pci: fix-up sleeping paths
ENS1370/1+ driver,ES1968 driver,Intel8x0 driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,ALI5451 driver,CS46xx driver
MIXART driver,RME HDSP driver,Trident driver,YMFPCI driver
Description: Fix-up sleeping in sound/pci. These changes fall under the
following two categories:
1) Replace schedule_timeout() with msleep() to guarantee the
task delays as expected. This also involved replacing/removing
custom sleep functions.
2) Do not assume jiffies will only increment by one if you
request a 1 jiffy sleep, i.e. use time_after/time_before in
while loops.
Signed-off-by: Nishanth Aravamudan <nacc@us.ibm.com>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2005-07-09 10:13:22 +02:00
end_time = jiffies + msecs_to_jiffies ( 750 ) ;
2005-04-16 15:20:36 -07:00
snd_via82xx_codec_xwrite ( chip , VIA_REG_AC97_READ |
VIA_REG_AC97_SECONDARY_VALID |
( VIA_REG_AC97_CODEC_ID_SECONDARY < < VIA_REG_AC97_CODEC_ID_SHIFT ) ) ;
do {
if ( ( val = snd_via82xx_codec_xread ( chip ) ) & VIA_REG_AC97_SECONDARY_VALID ) {
chip - > ac97_secondary = 1 ;
goto __ac97_ok2 ;
}
2005-10-24 15:02:37 +02:00
schedule_timeout_interruptible ( 1 ) ;
[ALSA] sound/pci: fix-up sleeping paths
ENS1370/1+ driver,ES1968 driver,Intel8x0 driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,ALI5451 driver,CS46xx driver
MIXART driver,RME HDSP driver,Trident driver,YMFPCI driver
Description: Fix-up sleeping in sound/pci. These changes fall under the
following two categories:
1) Replace schedule_timeout() with msleep() to guarantee the
task delays as expected. This also involved replacing/removing
custom sleep functions.
2) Do not assume jiffies will only increment by one if you
request a 1 jiffy sleep, i.e. use time_after/time_before in
while loops.
Signed-off-by: Nishanth Aravamudan <nacc@us.ibm.com>
Signed-off-by: Jaroslav Kysela <perex@suse.cz>
2005-07-09 10:13:22 +02:00
} while ( time_before ( jiffies , end_time ) ) ;
2005-04-16 15:20:36 -07:00
/* This is ok, the most of motherboards have only one codec */
__ac97_ok2 :
/* route FM trap to IRQ, disable FM trap */
// pci_write_config_byte(chip->pci, VIA_FM_NMI_CTRL, 0);
/* disable all GPI interrupts */
outl ( 0 , VIAREG ( chip , GPI_INTR ) ) ;
return 0 ;
}
# ifdef CONFIG_PM
/*
* power management
*/
2005-11-17 16:10:35 +01:00
static int snd_via82xx_suspend ( struct pci_dev * pci , pm_message_t state )
2005-04-16 15:20:36 -07:00
{
2005-11-17 16:10:35 +01:00
struct snd_card * card = pci_get_drvdata ( pci ) ;
struct via82xx_modem * chip = card - > private_data ;
2005-04-16 15:20:36 -07:00
int i ;
2005-11-17 16:10:35 +01:00
snd_power_change_state ( card , SNDRV_CTL_POWER_D3hot ) ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < 2 ; i + + )
2005-11-17 16:10:35 +01:00
snd_pcm_suspend_all ( chip - > pcms [ i ] ) ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < chip - > num_devs ; i + + )
snd_via82xx_channel_reset ( chip , & chip - > devs [ i ] ) ;
synchronize_irq ( chip - > irq ) ;
snd_ac97_suspend ( chip - > ac97 ) ;
2005-11-17 16:10:35 +01:00
pci_set_power_state ( pci , PCI_D3hot ) ;
pci_disable_device ( pci ) ;
pci_save_state ( pci ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2005-11-17 16:10:35 +01:00
static int snd_via82xx_resume ( struct pci_dev * pci )
2005-04-16 15:20:36 -07:00
{
2005-11-17 16:10:35 +01:00
struct snd_card * card = pci_get_drvdata ( pci ) ;
struct via82xx_modem * chip = card - > private_data ;
2005-04-16 15:20:36 -07:00
int i ;
2005-11-17 16:10:35 +01:00
pci_restore_state ( pci ) ;
pci_enable_device ( pci ) ;
pci_set_power_state ( pci , PCI_D0 ) ;
pci_set_master ( pci ) ;
2005-04-16 15:20:36 -07:00
snd_via82xx_chip_init ( chip ) ;
snd_ac97_resume ( chip - > ac97 ) ;
for ( i = 0 ; i < chip - > num_devs ; i + + )
snd_via82xx_channel_reset ( chip , & chip - > devs [ i ] ) ;
2005-11-17 16:10:35 +01:00
snd_power_change_state ( card , SNDRV_CTL_POWER_D0 ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
# endif /* CONFIG_PM */
2005-11-17 15:06:15 +01:00
static int snd_via82xx_free ( struct via82xx_modem * chip )
2005-04-16 15:20:36 -07:00
{
unsigned int i ;
if ( chip - > irq < 0 )
goto __end_hw ;
/* disable interrupts */
for ( i = 0 ; i < chip - > num_devs ; i + + )
snd_via82xx_channel_reset ( chip , & chip - > devs [ i ] ) ;
synchronize_irq ( chip - > irq ) ;
__end_hw :
if ( chip - > irq > = 0 )
2005-11-17 15:06:15 +01:00
free_irq ( chip - > irq , chip ) ;
2005-04-16 15:20:36 -07:00
pci_release_regions ( chip - > pci ) ;
pci_disable_device ( chip - > pci ) ;
kfree ( chip ) ;
return 0 ;
}
2005-11-17 15:06:15 +01:00
static int snd_via82xx_dev_free ( struct snd_device * device )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip = device - > device_data ;
2005-04-16 15:20:36 -07:00
return snd_via82xx_free ( chip ) ;
}
2005-11-17 15:06:15 +01:00
static int __devinit snd_via82xx_create ( struct snd_card * card ,
2005-04-16 15:20:36 -07:00
struct pci_dev * pci ,
int chip_type ,
int revision ,
unsigned int ac97_clock ,
2005-11-17 15:06:15 +01:00
struct via82xx_modem * * r_via )
2005-04-16 15:20:36 -07:00
{
2005-11-17 15:06:15 +01:00
struct via82xx_modem * chip ;
2005-04-16 15:20:36 -07:00
int err ;
2005-11-17 15:06:15 +01:00
static struct snd_device_ops ops = {
2005-04-16 15:20:36 -07:00
. dev_free = snd_via82xx_dev_free ,
} ;
if ( ( err = pci_enable_device ( pci ) ) < 0 )
return err ;
[ALSA] Replace with kzalloc() - pci stuff
AD1889 driver,ATIIXP driver,ATIIXP-modem driver,AZT3328 driver
BT87x driver,CMIPCI driver,CS4281 driver,ENS1370/1+ driver
ES1938 driver,ES1968 driver,FM801 driver,Intel8x0 driver
Intel8x0-modem driver,Maestro3 driver,SonicVibes driver,VIA82xx driver
VIA82xx-modem driver,AC97 Codec,AK4531 codec,au88x0 driver
CA0106 driver,CS46xx driver,EMU10K1/EMU10K2 driver,HDA Codec driver
HDA generic driver,HDA Intel driver,ICE1712 driver,ICE1724 driver
KORG1212 driver,MIXART driver,NM256 driver,Trident driver,YMFPCI driver
Replace kcalloc(1,..) with kzalloc().
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2005-09-09 14:21:46 +02:00
if ( ( chip = kzalloc ( sizeof ( * chip ) , GFP_KERNEL ) ) = = NULL ) {
2005-04-16 15:20:36 -07:00
pci_disable_device ( pci ) ;
return - ENOMEM ;
}
spin_lock_init ( & chip - > reg_lock ) ;
chip - > card = card ;
chip - > pci = pci ;
chip - > irq = - 1 ;
if ( ( err = pci_request_regions ( pci , card - > driver ) ) < 0 ) {
kfree ( chip ) ;
pci_disable_device ( pci ) ;
return err ;
}
chip - > port = pci_resource_start ( pci , 0 ) ;
if ( request_irq ( pci - > irq , snd_via82xx_interrupt , SA_INTERRUPT | SA_SHIRQ ,
2005-11-17 15:06:15 +01:00
card - > driver , chip ) ) {
2005-10-20 18:26:44 +02:00
snd_printk ( KERN_ERR " unable to grab IRQ %d \n " , pci - > irq ) ;
2005-04-16 15:20:36 -07:00
snd_via82xx_free ( chip ) ;
return - EBUSY ;
}
chip - > irq = pci - > irq ;
if ( ac97_clock > = 8000 & & ac97_clock < = 48000 )
chip - > ac97_clock = ac97_clock ;
synchronize_irq ( chip - > irq ) ;
if ( ( err = snd_via82xx_chip_init ( chip ) ) < 0 ) {
snd_via82xx_free ( chip ) ;
return err ;
}
if ( ( err = snd_device_new ( card , SNDRV_DEV_LOWLEVEL , chip , & ops ) ) < 0 ) {
snd_via82xx_free ( chip ) ;
return err ;
}
/* The 8233 ac97 controller does not implement the master bit
* in the pci command register . IMHO this is a violation of the PCI spec .
* We call pci_set_master here because it does not hurt . */
pci_set_master ( pci ) ;
snd_card_set_dev ( card , & pci - > dev ) ;
* r_via = chip ;
return 0 ;
}
static int __devinit snd_via82xx_probe ( struct pci_dev * pci ,
const struct pci_device_id * pci_id )
{
2005-11-17 15:06:15 +01:00
struct snd_card * card ;
struct via82xx_modem * chip ;
2005-04-16 15:20:36 -07:00
unsigned char revision ;
int chip_type = 0 , card_type ;
unsigned int i ;
int err ;
2005-10-04 08:46:51 +02:00
card = snd_card_new ( index , id , THIS_MODULE , 0 ) ;
2005-04-16 15:20:36 -07:00
if ( card = = NULL )
return - ENOMEM ;
card_type = pci_id - > driver_data ;
pci_read_config_byte ( pci , PCI_REVISION_ID , & revision ) ;
switch ( card_type ) {
case TYPE_CARD_VIA82XX_MODEM :
strcpy ( card - > driver , " VIA82XX-MODEM " ) ;
sprintf ( card - > shortname , " VIA 82XX modem " ) ;
break ;
default :
snd_printk ( KERN_ERR " invalid card type %d \n " , card_type ) ;
err = - EINVAL ;
goto __error ;
}
2005-10-04 08:46:51 +02:00
if ( ( err = snd_via82xx_create ( card , pci , chip_type , revision ,
ac97_clock , & chip ) ) < 0 )
2005-04-16 15:20:36 -07:00
goto __error ;
2005-11-17 16:10:35 +01:00
card - > private_data = chip ;
2005-04-16 15:20:36 -07:00
if ( ( err = snd_via82xx_mixer_new ( chip ) ) < 0 )
goto __error ;
if ( ( err = snd_via686_pcm_new ( chip ) ) < 0 )
goto __error ;
/* disable interrupts */
for ( i = 0 ; i < chip - > num_devs ; i + + )
snd_via82xx_channel_reset ( chip , & chip - > devs [ i ] ) ;
sprintf ( card - > longname , " %s at 0x%lx, irq %d " ,
card - > shortname , chip - > port , chip - > irq ) ;
snd_via82xx_proc_init ( chip ) ;
if ( ( err = snd_card_register ( card ) ) < 0 ) {
snd_card_free ( card ) ;
return err ;
}
pci_set_drvdata ( pci , card ) ;
return 0 ;
__error :
snd_card_free ( card ) ;
return err ;
}
static void __devexit snd_via82xx_remove ( struct pci_dev * pci )
{
snd_card_free ( pci_get_drvdata ( pci ) ) ;
pci_set_drvdata ( pci , NULL ) ;
}
static struct pci_driver driver = {
. name = " VIA 82xx Modem " ,
. id_table = snd_via82xx_modem_ids ,
. probe = snd_via82xx_probe ,
. remove = __devexit_p ( snd_via82xx_remove ) ,
2005-11-17 16:10:35 +01:00
# ifdef CONFIG_PM
. suspend = snd_via82xx_suspend ,
. resume = snd_via82xx_resume ,
# endif
2005-04-16 15:20:36 -07:00
} ;
static int __init alsa_card_via82xx_init ( void )
{
[ALSA] Replace pci_module_init() with pci_register_driver()
Documentation,ALS4000 driver,ATIIXP driver,ATIIXP-modem driver
AZT3328 driver,BT87x driver,CMIPCI driver,CS4281 driver
ENS1370/1+ driver,ES1938 driver,ES1968 driver,FM801 driver
Intel8x0 driver,Intel8x0-modem driver,Maestro3 driver,RME32 driver
RME96 driver,SonicVibes driver,VIA82xx driver,VIA82xx-modem driver
ALI5451 driver,au88x0 driver,CA0106 driver,CS46xx driver
EMU10K1/EMU10K2 driver,HDA Intel driver,ICE1712 driver,ICE1724 driver
KORG1212 driver,MIXART driver,NM256 driver,RME HDSP driver
RME9652 driver,Trident driver,Digigram VX222 driver,YMFPCI driver
Replace the obsolete pci_module_init() with pci_register_driver().
Signed-off-by: Takashi Iwai <tiwai@suse.de>
2005-04-11 16:58:24 +02:00
return pci_register_driver ( & driver ) ;
2005-04-16 15:20:36 -07:00
}
static void __exit alsa_card_via82xx_exit ( void )
{
pci_unregister_driver ( & driver ) ;
}
module_init ( alsa_card_via82xx_init )
module_exit ( alsa_card_via82xx_exit )