2005-04-16 15:20:36 -07:00
/*
* Driver for Digigram VX222 V2 / Mic PCI soundcards
*
* Copyright ( c ) 2002 by 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
*/
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/pci.h>
# include <linux/slab.h>
2011-07-15 13:13:37 -04:00
# include <linux/module.h>
2005-04-16 15:20:36 -07:00
# include <sound/core.h>
# include <sound/initval.h>
2006-08-23 19:53:28 +02:00
# include <sound/tlv.h>
2005-04-16 15:20:36 -07:00
# include "vx222.h"
# define CARD_NAME "VX222"
MODULE_AUTHOR ( " Takashi Iwai <tiwai@suse.de> " ) ;
MODULE_DESCRIPTION ( " Digigram VX222 V2/Mic " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_SUPPORTED_DEVICE ( " {{Digigram, " CARD_NAME " }} " ) ;
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 13:49:36 +10:30
static bool enable [ SNDRV_CARDS ] = SNDRV_DEFAULT_ENABLE_PNP ; /* Enable this card */
static bool mic [ SNDRV_CARDS ] ; /* microphone */
2005-04-16 15:20:36 -07:00
static int ibl [ SNDRV_CARDS ] ; /* microphone */
module_param_array ( index , int , NULL , 0444 ) ;
MODULE_PARM_DESC ( index , " Index value for Digigram " CARD_NAME " soundcard. " ) ;
module_param_array ( id , charp , NULL , 0444 ) ;
MODULE_PARM_DESC ( id , " ID string for Digigram " CARD_NAME " soundcard. " ) ;
module_param_array ( enable , bool , NULL , 0444 ) ;
MODULE_PARM_DESC ( enable , " Enable Digigram " CARD_NAME " soundcard. " ) ;
module_param_array ( mic , bool , NULL , 0444 ) ;
MODULE_PARM_DESC ( mic , " Enable Microphone. " ) ;
module_param_array ( ibl , int , NULL , 0444 ) ;
MODULE_PARM_DESC ( ibl , " Capture IBL size. " ) ;
/*
*/
enum {
VX_PCI_VX222_OLD ,
VX_PCI_VX222_NEW
} ;
2014-08-08 15:56:03 +02:00
static const struct pci_device_id snd_vx222_ids [ ] = {
2005-04-16 15:20:36 -07:00
{ 0x10b5 , 0x9050 , 0x1369 , PCI_ANY_ID , 0 , 0 , VX_PCI_VX222_OLD , } , /* PLX */
{ 0x10b5 , 0x9030 , 0x1369 , PCI_ANY_ID , 0 , 0 , VX_PCI_VX222_NEW , } , /* PLX */
{ 0 , }
} ;
MODULE_DEVICE_TABLE ( pci , snd_vx222_ids ) ;
/*
*/
2007-01-29 15:33:49 +01:00
static const DECLARE_TLV_DB_SCALE ( db_scale_old_vol , - 11350 , 50 , 0 ) ;
static const DECLARE_TLV_DB_SCALE ( db_scale_akm , - 7350 , 50 , 0 ) ;
2006-08-23 19:53:28 +02:00
2005-04-16 15:20:36 -07:00
static struct snd_vx_hardware vx222_old_hw = {
. name = " VX222/Old " ,
. type = VX_TYPE_BOARD ,
/* hw specs */
. num_codecs = 1 ,
. num_ins = 1 ,
. num_outs = 1 ,
. output_level_max = VX_ANALOG_OUT_LEVEL_MAX ,
2006-08-23 19:53:28 +02:00
. output_level_db_scale = db_scale_old_vol ,
2005-04-16 15:20:36 -07:00
} ;
static struct snd_vx_hardware vx222_v2_hw = {
. name = " VX222/v2 " ,
. type = VX_TYPE_V2 ,
/* hw specs */
. num_codecs = 1 ,
. num_ins = 1 ,
. num_outs = 1 ,
. output_level_max = VX2_AKM_LEVEL_MAX ,
2006-08-23 19:53:28 +02:00
. output_level_db_scale = db_scale_akm ,
2005-04-16 15:20:36 -07:00
} ;
static struct snd_vx_hardware vx222_mic_hw = {
. name = " VX222/Mic " ,
. type = VX_TYPE_MIC ,
/* hw specs */
. num_codecs = 1 ,
. num_ins = 1 ,
. num_outs = 1 ,
. output_level_max = VX2_AKM_LEVEL_MAX ,
2006-08-23 19:53:28 +02:00
. output_level_db_scale = db_scale_akm ,
2005-04-16 15:20:36 -07:00
} ;
/*
*/
2005-11-17 14:46:59 +01:00
static int snd_vx222_free ( struct vx_core * chip )
2005-04-16 15:20:36 -07:00
{
struct snd_vx222 * vx = ( struct snd_vx222 * ) chip ;
if ( chip - > irq > = 0 )
free_irq ( chip - > irq , ( void * ) chip ) ;
if ( vx - > port [ 0 ] )
pci_release_regions ( vx - > pci ) ;
pci_disable_device ( vx - > pci ) ;
kfree ( chip ) ;
return 0 ;
}
2005-11-17 14:46:59 +01:00
static int snd_vx222_dev_free ( struct snd_device * device )
2005-04-16 15:20:36 -07:00
{
2005-11-17 14:46:59 +01:00
struct vx_core * chip = device - > device_data ;
2005-04-16 15:20:36 -07:00
return snd_vx222_free ( chip ) ;
}
2012-12-06 12:35:10 -05:00
static int snd_vx222_create ( struct snd_card * card , struct pci_dev * pci ,
struct snd_vx_hardware * hw ,
struct snd_vx222 * * rchip )
2005-04-16 15:20:36 -07:00
{
2005-11-17 14:46:59 +01:00
struct vx_core * chip ;
2005-04-16 15:20:36 -07:00
struct snd_vx222 * vx ;
int i , err ;
2005-11-17 14:46:59 +01:00
static struct snd_device_ops ops = {
2005-04-16 15:20:36 -07:00
. dev_free = snd_vx222_dev_free ,
} ;
struct snd_vx_ops * vx_ops ;
/* enable PCI device */
if ( ( err = pci_enable_device ( pci ) ) < 0 )
return err ;
pci_set_master ( pci ) ;
vx_ops = hw - > type = = VX_TYPE_BOARD ? & vx222_old_ops : & vx222_ops ;
chip = snd_vx_create ( card , hw , vx_ops ,
2005-11-17 14:46:59 +01:00
sizeof ( struct snd_vx222 ) - sizeof ( struct vx_core ) ) ;
2005-04-16 15:20:36 -07:00
if ( ! chip ) {
pci_disable_device ( pci ) ;
return - ENOMEM ;
}
vx = ( struct snd_vx222 * ) chip ;
vx - > pci = pci ;
if ( ( err = pci_request_regions ( pci , CARD_NAME ) ) < 0 ) {
snd_vx222_free ( chip ) ;
return err ;
}
for ( i = 0 ; i < 2 ; i + + )
vx - > port [ i ] = pci_resource_start ( pci , i + 1 ) ;
2014-09-09 17:17:20 +02:00
if ( request_threaded_irq ( pci - > irq , snd_vx_irq_handler ,
snd_vx_threaded_irq_handler , IRQF_SHARED ,
KBUILD_MODNAME , chip ) ) {
2014-02-26 12:13:57 +01:00
dev_err ( card - > dev , " unable to grab IRQ %d \n " , pci - > irq ) ;
2005-04-16 15:20:36 -07:00
snd_vx222_free ( chip ) ;
return - EBUSY ;
}
chip - > irq = pci - > irq ;
if ( ( err = snd_device_new ( card , SNDRV_DEV_LOWLEVEL , chip , & ops ) ) < 0 ) {
snd_vx222_free ( chip ) ;
return err ;
}
* rchip = vx ;
return 0 ;
}
2012-12-06 12:35:10 -05:00
static int snd_vx222_probe ( struct pci_dev * pci ,
const struct pci_device_id * pci_id )
2005-04-16 15:20:36 -07:00
{
static int dev ;
2005-11-17 14:46:59 +01:00
struct snd_card * card ;
2005-04-16 15:20:36 -07:00
struct snd_vx_hardware * hw ;
struct snd_vx222 * vx ;
int err ;
if ( dev > = SNDRV_CARDS )
return - ENODEV ;
if ( ! enable [ dev ] ) {
dev + + ;
return - ENOENT ;
}
2014-01-29 14:20:19 +01:00
err = snd_card_new ( & pci - > dev , index [ dev ] , id [ dev ] , THIS_MODULE ,
0 , & card ) ;
2008-12-28 16:44:30 +01:00
if ( err < 0 )
return err ;
2005-04-16 15:20:36 -07:00
switch ( ( int ) pci_id - > driver_data ) {
case VX_PCI_VX222_OLD :
hw = & vx222_old_hw ;
break ;
case VX_PCI_VX222_NEW :
default :
if ( mic [ dev ] )
hw = & vx222_mic_hw ;
else
hw = & vx222_v2_hw ;
break ;
}
if ( ( err = snd_vx222_create ( card , pci , hw , & vx ) ) < 0 ) {
snd_card_free ( card ) ;
return err ;
}
2005-11-17 16:06:05 +01:00
card - > private_data = vx ;
2005-04-16 15:20:36 -07:00
vx - > core . ibl . size = ibl [ dev ] ;
sprintf ( card - > longname , " %s at 0x%lx & 0x%lx, irq %i " ,
card - > shortname , vx - > port [ 0 ] , vx - > port [ 1 ] , vx - > core . irq ) ;
2014-02-26 12:13:57 +01:00
dev_dbg ( card - > dev , " %s at 0x%lx & 0x%lx, irq %i \n " ,
2005-04-16 15:20:36 -07:00
card - > shortname , vx - > port [ 0 ] , vx - > port [ 1 ] , vx - > core . irq ) ;
# ifdef SND_VX_FW_LOADER
vx - > core . dev = & pci - > dev ;
# endif
if ( ( err = snd_vx_setup_firmware ( & vx - > core ) ) < 0 ) {
snd_card_free ( card ) ;
return err ;
}
if ( ( err = snd_card_register ( card ) ) < 0 ) {
snd_card_free ( card ) ;
return err ;
}
pci_set_drvdata ( pci , card ) ;
dev + + ;
return 0 ;
}
2012-12-06 12:35:10 -05:00
static void snd_vx222_remove ( struct pci_dev * pci )
2005-04-16 15:20:36 -07:00
{
snd_card_free ( pci_get_drvdata ( pci ) ) ;
}
2012-08-14 18:12:04 +02:00
# ifdef CONFIG_PM_SLEEP
2012-07-02 15:20:37 +02:00
static int snd_vx222_suspend ( struct device * dev )
2005-11-17 16:06:05 +01:00
{
2012-07-02 15:20:37 +02:00
struct snd_card * card = dev_get_drvdata ( dev ) ;
2005-11-17 16:06:05 +01:00
struct snd_vx222 * vx = card - > private_data ;
2015-01-08 12:50:48 +01:00
return snd_vx_suspend ( & vx - > core ) ;
2005-11-17 16:06:05 +01:00
}
2012-07-02 15:20:37 +02:00
static int snd_vx222_resume ( struct device * dev )
2005-11-17 16:06:05 +01:00
{
2012-07-02 15:20:37 +02:00
struct snd_card * card = dev_get_drvdata ( dev ) ;
2005-11-17 16:06:05 +01:00
struct snd_vx222 * vx = card - > private_data ;
return snd_vx_resume ( & vx - > core ) ;
}
2012-07-02 15:20:37 +02:00
static SIMPLE_DEV_PM_OPS ( snd_vx222_pm , snd_vx222_suspend , snd_vx222_resume ) ;
# define SND_VX222_PM_OPS &snd_vx222_pm
# else
# define SND_VX222_PM_OPS NULL
2005-11-17 16:06:05 +01:00
# endif
2012-04-24 12:25:00 +02:00
static struct pci_driver vx222_driver = {
2011-06-10 16:20:20 +02:00
. name = KBUILD_MODNAME ,
2005-04-16 15:20:36 -07:00
. id_table = snd_vx222_ids ,
. probe = snd_vx222_probe ,
2012-12-06 12:35:10 -05:00
. remove = snd_vx222_remove ,
2012-07-02 15:20:37 +02:00
. driver = {
. pm = SND_VX222_PM_OPS ,
} ,
2005-04-16 15:20:36 -07:00
} ;
2012-04-24 12:25:00 +02:00
module_pci_driver ( vx222_driver ) ;