2005-04-16 15:20:36 -07:00
/*
*
2007-07-27 16:52:19 +02:00
* hda_intel . c - Implementation of primary alsa driver code base
* for Intel HD Audio .
2005-04-16 15:20:36 -07:00
*
* Copyright ( c ) 2004 Intel Corporation . All rights reserved .
*
* Copyright ( c ) 2004 Takashi Iwai < tiwai @ suse . de >
* PeiSen Hou < pshou @ realtek . com . tw >
*
* 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 .
*
* CONTACTS :
*
* Matt Jared matt . jared @ intel . com
* Andy Kopp andy . kopp @ intel . com
* Dan Kogan dan . d . kogan @ intel . com
*
* CHANGES :
*
* 2004.12 .01 Major rewrite by tiwai , merged the work of pshou
*
*/
# include <linux/delay.h>
# include <linux/interrupt.h>
2005-11-07 14:43:23 +01:00
# include <linux/kernel.h>
2005-04-16 15:20:36 -07:00
# include <linux/module.h>
2008-03-04 10:08:58 +01:00
# include <linux/dma-mapping.h>
2005-04-16 15:20:36 -07:00
# include <linux/moduleparam.h>
# include <linux/init.h>
# include <linux/slab.h>
# include <linux/pci.h>
2006-01-16 16:34:20 +01:00
# include <linux/mutex.h>
2011-09-28 17:16:09 +02:00
# include <linux/io.h>
2012-08-23 17:32:30 +08:00
# include <linux/pm_runtime.h>
2012-10-22 16:42:16 -05:00
# include <linux/clocksource.h>
# include <linux/time.h>
2012-12-04 15:09:23 +01:00
# include <linux/completion.h>
2012-10-22 16:42:16 -05:00
2011-09-28 17:16:09 +02:00
# ifdef CONFIG_X86
/* for snoop control */
# include <asm/pgtable.h>
# include <asm/cacheflush.h>
# endif
2005-04-16 15:20:36 -07:00
# include <sound/core.h>
# include <sound/initval.h>
2012-04-26 12:13:25 +02:00
# include <linux/vgaarb.h>
2012-04-26 12:23:42 +02:00
# include <linux/vga_switcheroo.h>
2012-08-09 12:33:28 +02:00
# include <linux/firmware.h>
2005-04-16 15:20:36 -07:00
# include "hda_codec.h"
2014-02-28 15:41:22 -08:00
# include "hda_controller.h"
2014-02-28 15:41:12 -08:00
# include "hda_priv.h"
2014-07-03 17:02:23 +08:00
# include "hda_i915.h"
2005-04-16 15:20:36 -07:00
2014-06-26 16:50:16 +02:00
/* position fix mode */
enum {
POS_FIX_AUTO ,
POS_FIX_LPIB ,
POS_FIX_POSBUF ,
POS_FIX_VIACOMBO ,
POS_FIX_COMBO ,
} ;
2014-06-26 17:19:20 +02:00
/* Defines for ATI HD Audio support in SB450 south bridge */
# define ATI_SB450_HDAUDIO_MISC_CNTR2_ADDR 0x42
# define ATI_SB450_HDAUDIO_ENABLE_SNOOP 0x02
/* Defines for Nvidia HDA support */
# define NVIDIA_HDA_TRANSREG_ADDR 0x4e
# define NVIDIA_HDA_ENABLE_COHBITS 0x0f
# define NVIDIA_HDA_ISTRM_COH 0x4d
# define NVIDIA_HDA_OSTRM_COH 0x4c
# define NVIDIA_HDA_ENABLE_COHBIT 0x01
/* Defines for Intel SCH HDA snoop control */
# define INTEL_SCH_HDA_DEVC 0x78
# define INTEL_SCH_HDA_DEVC_NOSNOOP (0x1<<11)
/* Define IN stream 0 FIFO size offset in VIA controller */
# define VIA_IN_STREAM0_FIFO_SIZE_OFFSET 0x90
/* Define VIA HD Audio Device ID*/
# define VIA_HDAC_DEVICE_ID 0x3288
2014-06-26 17:28:06 +02:00
/* max number of SDs */
/* ICH, ATI and VIA have 4 playback and 4 capture */
# define ICH6_NUM_CAPTURE 4
# define ICH6_NUM_PLAYBACK 4
/* ULI has 6 playback and 5 capture */
# define ULI_NUM_CAPTURE 5
# define ULI_NUM_PLAYBACK 6
/* ATI HDMI may have up to 8 playbacks and 0 capture */
# define ATIHDMI_NUM_CAPTURE 0
# define ATIHDMI_NUM_PLAYBACK 8
/* TERA has 4 playback and 3 capture */
# define TERA_NUM_CAPTURE 3
# define TERA_NUM_PLAYBACK 4
2005-04-16 15:20:36 -07:00
2008-01-07 15:16:37 +01:00
static int index [ SNDRV_CARDS ] = SNDRV_DEFAULT_IDX ;
static char * id [ SNDRV_CARDS ] = SNDRV_DEFAULT_STR ;
2011-12-15 13:49:36 +10:30
static bool enable [ SNDRV_CARDS ] = SNDRV_DEFAULT_ENABLE_PNP ;
2008-01-07 15:16:37 +01:00
static char * model [ SNDRV_CARDS ] ;
2012-09-13 14:59:47 +02:00
static int position_fix [ SNDRV_CARDS ] = { [ 0 . . . ( SNDRV_CARDS - 1 ) ] = - 1 } ;
2008-06-10 17:53:35 +02:00
static int bdl_pos_adj [ SNDRV_CARDS ] = { [ 0 . . . ( SNDRV_CARDS - 1 ) ] = - 1 } ;
2008-01-07 15:16:37 +01:00
static int probe_mask [ SNDRV_CARDS ] = { [ 0 . . . ( SNDRV_CARDS - 1 ) ] = - 1 } ;
2008-12-19 15:19:11 +01:00
static int probe_only [ SNDRV_CARDS ] ;
2012-10-09 15:04:21 +02:00
static int jackpoll_ms [ SNDRV_CARDS ] ;
2011-12-15 13:49:36 +10:30
static bool single_cmd ;
2009-09-28 13:14:04 +02:00
static int enable_msi = - 1 ;
2009-06-17 09:52:54 +02:00
# ifdef CONFIG_SND_HDA_PATCH_LOADER
static char * patch [ SNDRV_CARDS ] ;
# endif
2009-11-13 18:41:52 +01:00
# ifdef CONFIG_SND_HDA_INPUT_BEEP
2012-07-03 16:58:48 +02:00
static bool beep_mode [ SNDRV_CARDS ] = { [ 0 . . . ( SNDRV_CARDS - 1 ) ] =
2009-11-13 18:41:52 +01:00
CONFIG_SND_HDA_INPUT_BEEP_MODE } ;
# endif
2005-04-16 15:20:36 -07:00
2008-01-07 15:16:37 +01:00
module_param_array ( index , int , NULL , 0444 ) ;
2005-04-16 15:20:36 -07:00
MODULE_PARM_DESC ( index , " Index value for Intel HD audio interface. " ) ;
2008-01-07 15:16:37 +01:00
module_param_array ( id , charp , NULL , 0444 ) ;
2005-04-16 15:20:36 -07:00
MODULE_PARM_DESC ( id , " ID string for Intel HD audio interface. " ) ;
2008-01-07 15:16:37 +01:00
module_param_array ( enable , bool , NULL , 0444 ) ;
MODULE_PARM_DESC ( enable , " Enable Intel HD audio interface. " ) ;
module_param_array ( model , charp , NULL , 0444 ) ;
2005-04-16 15:20:36 -07:00
MODULE_PARM_DESC ( model , " Use the given board model. " ) ;
2008-01-07 15:16:37 +01:00
module_param_array ( position_fix , int , NULL , 0444 ) ;
2010-09-30 10:12:50 +02:00
MODULE_PARM_DESC ( position_fix , " DMA pointer read method. "
2012-09-13 14:59:47 +02:00
" (-1 = system default, 0 = auto, 1 = LPIB, 2 = POSBUF, 3 = VIACOMBO, 4 = COMBO). " ) ;
2008-06-10 17:53:34 +02:00
module_param_array ( bdl_pos_adj , int , NULL , 0644 ) ;
MODULE_PARM_DESC ( bdl_pos_adj , " BDL position adjustment offset. " ) ;
2008-01-07 15:16:37 +01:00
module_param_array ( probe_mask , int , NULL , 0444 ) ;
2005-11-24 16:03:40 +01:00
MODULE_PARM_DESC ( probe_mask , " Bitmask to probe codecs (default = -1). " ) ;
2010-03-26 11:16:59 +01:00
module_param_array ( probe_only , int , NULL , 0444 ) ;
2008-12-19 15:19:11 +01:00
MODULE_PARM_DESC ( probe_only , " Only probing and no codec initialization. " ) ;
2012-10-09 15:04:21 +02:00
module_param_array ( jackpoll_ms , int , NULL , 0444 ) ;
MODULE_PARM_DESC ( jackpoll_ms , " Ms between polling for jack events (default = 0, using unsol events only) " ) ;
2006-01-12 18:28:44 +01:00
module_param ( single_cmd , bool , 0444 ) ;
2007-07-27 16:52:19 +02:00
MODULE_PARM_DESC ( single_cmd , " Use single command to communicate with codecs "
" (for debugging only). " ) ;
2012-01-20 12:08:44 +01:00
module_param ( enable_msi , bint , 0444 ) ;
2006-11-10 12:08:37 +01:00
MODULE_PARM_DESC ( enable_msi , " Enable Message Signaled Interrupt (MSI) " ) ;
2009-06-17 09:52:54 +02:00
# ifdef CONFIG_SND_HDA_PATCH_LOADER
module_param_array ( patch , charp , NULL , 0444 ) ;
MODULE_PARM_DESC ( patch , " Patch file for Intel HD audio interface. " ) ;
# endif
2009-11-13 18:41:52 +01:00
# ifdef CONFIG_SND_HDA_INPUT_BEEP
2012-07-03 16:58:48 +02:00
module_param_array ( beep_mode , bool , NULL , 0444 ) ;
2009-11-13 18:41:52 +01:00
MODULE_PARM_DESC ( beep_mode , " Select HDA Beep registration mode "
2012-07-03 16:58:48 +02:00
" (0=off, 1=on) (default=1). " ) ;
2009-11-13 18:41:52 +01:00
# endif
2005-11-24 16:03:40 +01:00
2012-08-24 18:38:08 +02:00
# ifdef CONFIG_PM
2012-08-14 17:13:32 +02:00
static int param_set_xint ( const char * val , const struct kernel_param * kp ) ;
static struct kernel_param_ops param_ops_xint = {
. set = param_set_xint ,
. get = param_get_int ,
} ;
# define param_check_xint param_check_int
2008-11-27 12:43:28 +01:00
static int power_save = CONFIG_SND_HDA_POWER_SAVE_DEFAULT ;
2014-02-28 15:41:19 -08:00
static int * power_save_addr = & power_save ;
2012-08-14 17:13:32 +02:00
module_param ( power_save , xint , 0644 ) ;
2008-11-27 12:43:28 +01:00
MODULE_PARM_DESC ( power_save , " Automatic power-saving timeout "
" (in second, 0 = disable). " ) ;
2005-04-16 15:20:36 -07:00
2007-08-13 16:10:30 +02:00
/* reset the HD-audio controller in power save mode.
* this may give more power - saving , but will take longer time to
* wake up .
*/
2013-04-04 15:35:24 +02:00
static bool power_save_controller = 1 ;
module_param ( power_save_controller , bool , 0644 ) ;
2007-08-13 16:10:30 +02:00
MODULE_PARM_DESC ( power_save_controller , " Reset controller in power save mode. " ) ;
2014-02-28 15:41:19 -08:00
# else
static int * power_save_addr ;
2012-08-24 18:38:08 +02:00
# endif /* CONFIG_PM */
2007-08-13 16:10:30 +02:00
2012-01-23 17:53:39 +01:00
static int align_buffer_size = - 1 ;
module_param ( align_buffer_size , bint , 0644 ) ;
2011-08-04 10:12:56 -05:00
MODULE_PARM_DESC ( align_buffer_size ,
" Force buffer and period sizes to be multiple of 128 bytes. " ) ;
2011-09-28 17:16:09 +02:00
# ifdef CONFIG_X86
static bool hda_snoop = true ;
module_param_named ( snoop , hda_snoop , bool , 0444 ) ;
MODULE_PARM_DESC ( snoop , " Enable/disable snooping " ) ;
# else
# define hda_snoop true
# endif
2005-04-16 15:20:36 -07:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_SUPPORTED_DEVICE ( " {{Intel, ICH6}, "
" {Intel, ICH6M}, "
2005-05-01 08:58:50 -07:00
" {Intel, ICH7}, "
2005-05-12 14:55:20 +02:00
" {Intel, ESB2}, "
2006-01-10 11:07:37 +01:00
" {Intel, ICH8}, "
2006-11-22 11:53:52 +01:00
" {Intel, ICH9}, "
2008-01-29 12:38:49 +01:00
" {Intel, ICH10}, "
2008-08-08 15:56:39 -07:00
" {Intel, PCH}, "
2010-01-12 17:03:35 -08:00
" {Intel, CPT}, "
2011-04-20 10:59:57 -07:00
" {Intel, PPT}, "
2012-01-23 16:24:31 -08:00
" {Intel, LPT}, "
2012-08-09 09:38:59 -07:00
" {Intel, LPT_LP}, "
2013-11-04 09:27:45 -08:00
" {Intel, WPT_LP}, "
2012-06-13 10:23:51 +08:00
" {Intel, HPT}, "
2010-09-10 16:29:56 -07:00
" {Intel, PBG}, "
2008-01-30 08:13:55 +01:00
" {Intel, SCH}, "
2005-05-12 15:00:41 +02:00
" {ATI, SB450}, "
2006-03-31 12:33:59 +02:00
" {ATI, SB600}, "
2006-05-17 11:22:21 +02:00
" {ATI, RS600}, "
2006-10-16 12:49:47 +02:00
" {ATI, RS690}, "
2007-04-27 12:20:57 +02:00
" {ATI, RS780}, "
" {ATI, R600}, "
2007-11-05 18:21:56 +01:00
" {ATI, RV630}, "
" {ATI, RV610}, "
2007-11-16 11:06:30 +01:00
" {ATI, RV670}, "
" {ATI, RV635}, "
" {ATI, RV620}, "
" {ATI, RV770}, "
2005-05-12 15:00:41 +02:00
" {VIA, VT8251}, "
2005-08-12 16:44:04 +02:00
" {VIA, VT8237A}, "
2005-08-24 14:14:57 +02:00
" {SiS, SIS966}, "
" {ULI, M5461}} " ) ;
2005-04-16 15:20:36 -07:00
MODULE_DESCRIPTION ( " Intel HDA driver " ) ;
2012-04-26 12:23:42 +02:00
# if defined(CONFIG_PM) && defined(CONFIG_VGA_SWITCHEROO)
2014-02-06 18:14:03 +01:00
# if IS_ENABLED(CONFIG_SND_HDA_CODEC_HDMI)
2012-04-26 12:23:42 +02:00
# define SUPPORT_VGA_SWITCHEROO
# endif
# endif
2005-04-16 15:20:36 -07:00
/*
*/
2005-08-24 14:14:57 +02:00
/* driver types */
enum {
AZX_DRIVER_ICH ,
2010-02-22 17:31:09 -08:00
AZX_DRIVER_PCH ,
2008-01-30 08:13:55 +01:00
AZX_DRIVER_SCH ,
2013-11-05 17:54:05 +01:00
AZX_DRIVER_HDMI ,
2005-08-24 14:14:57 +02:00
AZX_DRIVER_ATI ,
2006-05-17 11:22:21 +02:00
AZX_DRIVER_ATIHDMI ,
2011-12-14 16:10:27 +08:00
AZX_DRIVER_ATIHDMI_NS ,
2005-08-24 14:14:57 +02:00
AZX_DRIVER_VIA ,
AZX_DRIVER_SIS ,
AZX_DRIVER_ULI ,
2005-09-13 18:49:12 +02:00
AZX_DRIVER_NVIDIA ,
2008-05-27 11:44:55 +02:00
AZX_DRIVER_TERA ,
2010-10-21 09:03:25 +02:00
AZX_DRIVER_CTX ,
2012-05-08 10:34:08 +02:00
AZX_DRIVER_CTHDA ,
2014-08-06 14:27:42 +02:00
AZX_DRIVER_CMEDIA ,
2008-11-13 11:07:07 +01:00
AZX_DRIVER_GENERIC ,
2008-09-03 16:00:44 +02:00
AZX_NUM_DRIVERS , /* keep this as last entry */
2005-08-24 14:14:57 +02:00
} ;
2012-11-19 20:03:37 +01:00
/* quirks for Intel PCH */
2013-01-08 13:51:30 +01:00
# define AZX_DCAPS_INTEL_PCH_NOPM \
2012-11-19 20:03:37 +01:00
( AZX_DCAPS_SCH_SNOOP | AZX_DCAPS_BUFSIZE | \
2014-07-14 10:45:31 +02:00
AZX_DCAPS_COUNT_LPIB_DELAY | AZX_DCAPS_REVERSE_ASSIGN )
2013-01-08 13:51:30 +01:00
# define AZX_DCAPS_INTEL_PCH \
( AZX_DCAPS_INTEL_PCH_NOPM | AZX_DCAPS_PM_RUNTIME )
2011-05-25 09:11:37 +02:00
2013-11-05 17:34:46 +01:00
# define AZX_DCAPS_INTEL_HASWELL \
( AZX_DCAPS_SCH_SNOOP | AZX_DCAPS_ALIGN_BUFSIZE | \
AZX_DCAPS_COUNT_LPIB_DELAY | AZX_DCAPS_PM_RUNTIME | \
AZX_DCAPS_I915_POWERWELL )
2014-06-09 15:28:59 +08:00
/* Broadwell HDMI can't use position buffer reliably, force to use LPIB */
# define AZX_DCAPS_INTEL_BROADWELL \
( AZX_DCAPS_SCH_SNOOP | AZX_DCAPS_ALIGN_BUFSIZE | \
AZX_DCAPS_POSFIX_LPIB | AZX_DCAPS_PM_RUNTIME | \
AZX_DCAPS_I915_POWERWELL )
2011-05-25 09:11:37 +02:00
/* quirks for ATI SB / AMD Hudson */
# define AZX_DCAPS_PRESET_ATI_SB \
( AZX_DCAPS_ATI_SNOOP | AZX_DCAPS_NO_TCSEL | \
AZX_DCAPS_SYNC_WRITE | AZX_DCAPS_POSFIX_LPIB )
/* quirks for ATI/AMD HDMI */
# define AZX_DCAPS_PRESET_ATI_HDMI \
( AZX_DCAPS_NO_TCSEL | AZX_DCAPS_SYNC_WRITE | AZX_DCAPS_POSFIX_LPIB )
/* quirks for Nvidia */
# define AZX_DCAPS_PRESET_NVIDIA \
2012-01-23 17:53:39 +01:00
( AZX_DCAPS_NVIDIA_SNOOP | AZX_DCAPS_RIRB_DELAY | AZX_DCAPS_NO_MSI | \
2014-04-29 18:38:21 +02:00
AZX_DCAPS_ALIGN_BUFSIZE | AZX_DCAPS_NO_64BIT | \
AZX_DCAPS_CORBRP_SELF_CLEAR )
2011-05-25 09:11:37 +02:00
2012-05-08 10:34:08 +02:00
# define AZX_DCAPS_PRESET_CTHDA \
( AZX_DCAPS_NO_MSI | AZX_DCAPS_POSFIX_LPIB | AZX_DCAPS_4K_BDLE_BOUNDARY )
2012-04-26 12:23:42 +02:00
/*
* VGA - switcher support
*/
# ifdef SUPPORT_VGA_SWITCHEROO
2012-08-09 13:49:23 +02:00
# define use_vga_switcheroo(chip) ((chip)->use_vga_switcheroo)
# else
# define use_vga_switcheroo(chip) 0
# endif
2012-12-07 07:40:35 +01:00
static char * driver_short_names [ ] = {
2005-08-24 14:14:57 +02:00
[ AZX_DRIVER_ICH ] = " HDA Intel " ,
2010-02-22 17:31:09 -08:00
[ AZX_DRIVER_PCH ] = " HDA Intel PCH " ,
2008-01-30 08:13:55 +01:00
[ AZX_DRIVER_SCH ] = " HDA Intel MID " ,
2013-11-05 17:54:05 +01:00
[ AZX_DRIVER_HDMI ] = " HDA Intel HDMI " ,
2005-08-24 14:14:57 +02:00
[ AZX_DRIVER_ATI ] = " HDA ATI SB " ,
2006-05-17 11:22:21 +02:00
[ AZX_DRIVER_ATIHDMI ] = " HDA ATI HDMI " ,
2011-12-14 16:10:27 +08:00
[ AZX_DRIVER_ATIHDMI_NS ] = " HDA ATI HDMI " ,
2005-08-24 14:14:57 +02:00
[ AZX_DRIVER_VIA ] = " HDA VIA VT82xx " ,
[ AZX_DRIVER_SIS ] = " HDA SIS966 " ,
2005-09-13 18:49:12 +02:00
[ AZX_DRIVER_ULI ] = " HDA ULI M5461 " ,
[ AZX_DRIVER_NVIDIA ] = " HDA NVidia " ,
2008-05-27 11:44:55 +02:00
[ AZX_DRIVER_TERA ] = " HDA Teradici " ,
2010-10-21 09:03:25 +02:00
[ AZX_DRIVER_CTX ] = " HDA Creative " ,
2012-05-08 10:34:08 +02:00
[ AZX_DRIVER_CTHDA ] = " HDA Creative " ,
2014-08-06 14:27:42 +02:00
[ AZX_DRIVER_CMEDIA ] = " HDA C-Media " ,
2008-11-13 11:07:07 +01:00
[ AZX_DRIVER_GENERIC ] = " HD-Audio Generic " ,
2005-08-24 14:14:57 +02:00
} ;
2014-06-26 18:45:16 +08:00
struct hda_intel {
struct azx chip ;
2014-06-26 17:19:20 +02:00
/* for pending irqs */
struct work_struct irq_pending_work ;
/* sync probing */
struct completion probe_wait ;
struct work_struct probe_work ;
/* card list (for power_save trigger) */
struct list_head list ;
/* extra flags */
unsigned int irq_pending_warned : 1 ;
/* VGA-switcheroo setup */
unsigned int use_vga_switcheroo : 1 ;
unsigned int vga_switcheroo_registered : 1 ;
unsigned int init_failed : 1 ; /* delayed init failed */
/* secondary power domain for hdmi audio under vga device */
struct dev_pm_domain hdmi_pm_domain ;
} ;
2014-06-26 18:45:16 +08:00
2011-09-28 17:16:09 +02:00
# ifdef CONFIG_X86
2013-01-29 18:07:22 +01:00
static void __mark_pages_wc ( struct azx * chip , struct snd_dma_buffer * dmab , bool on )
2011-09-28 17:16:09 +02:00
{
2013-01-29 18:07:22 +01:00
int pages ;
2011-09-28 17:16:09 +02:00
if ( azx_snoop ( chip ) )
return ;
2013-01-29 18:07:22 +01:00
if ( ! dmab | | ! dmab - > area | | ! dmab - > bytes )
return ;
# ifdef CONFIG_SND_DMA_SGBUF
if ( dmab - > dev . type = = SNDRV_DMA_TYPE_DEV_SG ) {
struct snd_sg_buf * sgbuf = dmab - > private_data ;
2011-09-28 17:16:09 +02:00
if ( on )
2013-01-29 18:07:22 +01:00
set_pages_array_wc ( sgbuf - > page_table , sgbuf - > pages ) ;
2011-09-28 17:16:09 +02:00
else
2013-01-29 18:07:22 +01:00
set_pages_array_wb ( sgbuf - > page_table , sgbuf - > pages ) ;
return ;
2011-09-28 17:16:09 +02:00
}
2013-01-29 18:07:22 +01:00
# endif
pages = ( dmab - > bytes + PAGE_SIZE - 1 ) > > PAGE_SHIFT ;
if ( on )
set_memory_wc ( ( unsigned long ) dmab - > area , pages ) ;
else
set_memory_wb ( ( unsigned long ) dmab - > area , pages ) ;
2011-09-28 17:16:09 +02:00
}
static inline void mark_pages_wc ( struct azx * chip , struct snd_dma_buffer * buf ,
bool on )
{
2013-01-29 18:07:22 +01:00
__mark_pages_wc ( chip , buf , on ) ;
2011-09-28 17:16:09 +02:00
}
static inline void mark_runtime_wc ( struct azx * chip , struct azx_dev * azx_dev ,
2013-01-29 18:07:22 +01:00
struct snd_pcm_substream * substream , bool on )
2011-09-28 17:16:09 +02:00
{
if ( azx_dev - > wc_marked ! = on ) {
2013-01-29 18:07:22 +01:00
__mark_pages_wc ( chip , snd_pcm_get_dma_buf ( substream ) , on ) ;
2011-09-28 17:16:09 +02:00
azx_dev - > wc_marked = on ;
}
}
# else
/* NOP for other archs */
static inline void mark_pages_wc ( struct azx * chip , struct snd_dma_buffer * buf ,
bool on )
{
}
static inline void mark_runtime_wc ( struct azx * chip , struct azx_dev * azx_dev ,
2013-01-29 18:07:22 +01:00
struct snd_pcm_substream * substream , bool on )
2011-09-28 17:16:09 +02:00
{
}
# endif
2006-10-23 13:40:59 +02:00
static int azx_acquire_irq ( struct azx * chip , int do_disconnect ) ;
2006-02-16 18:17:58 +01:00
2007-08-10 17:21:45 +02:00
/*
* initialize the PCI registers
*/
/* update bits in a PCI register byte */
static void update_pci_byte ( struct pci_dev * pci , unsigned int reg ,
unsigned char mask , unsigned char val )
{
unsigned char data ;
pci_read_config_byte ( pci , reg , & data ) ;
data & = ~ mask ;
data | = ( val & mask ) ;
pci_write_config_byte ( pci , reg , data ) ;
}
static void azx_init_pci ( struct azx * chip )
{
/* Clear bits 0-2 of PCI register TCSEL (at offset 0x44)
* TCSEL = = Traffic Class Select Register , which sets PCI express QOS
* Ensuring these bits are 0 clears playback static on some HD Audio
2011-03-10 17:41:56 +01:00
* codecs .
* The PCI register TCSEL is defined in the Intel manuals .
2007-08-10 17:21:45 +02:00
*/
2011-05-27 19:45:28 -07:00
if ( ! ( chip - > driver_caps & AZX_DCAPS_NO_TCSEL ) ) {
2014-02-25 12:21:03 +01:00
dev_dbg ( chip - > card - > dev , " Clearing TCSEL \n " ) ;
2014-06-26 17:54:37 +02:00
update_pci_byte ( chip - > pci , AZX_PCIREG_TCSEL , 0x07 , 0 ) ;
2011-05-25 09:11:37 +02:00
}
2007-08-10 17:21:45 +02:00
2011-05-25 09:11:37 +02:00
/* For ATI SB450/600/700/800/900 and AMD Hudson azalia HD audio,
* we need to enable snoop .
*/
if ( chip - > driver_caps & AZX_DCAPS_ATI_SNOOP ) {
2014-02-25 12:21:03 +01:00
dev_dbg ( chip - > card - > dev , " Setting ATI snoop: %d \n " ,
azx_snoop ( chip ) ) ;
2007-08-10 17:21:45 +02:00
update_pci_byte ( chip - > pci ,
2011-09-28 17:16:09 +02:00
ATI_SB450_HDAUDIO_MISC_CNTR2_ADDR , 0x07 ,
azx_snoop ( chip ) ? ATI_SB450_HDAUDIO_ENABLE_SNOOP : 0 ) ;
2011-05-25 09:11:37 +02:00
}
/* For NVIDIA HDA, enable snoop */
if ( chip - > driver_caps & AZX_DCAPS_NVIDIA_SNOOP ) {
2014-02-25 12:21:03 +01:00
dev_dbg ( chip - > card - > dev , " Setting Nvidia snoop: %d \n " ,
azx_snoop ( chip ) ) ;
2007-08-10 17:21:45 +02:00
update_pci_byte ( chip - > pci ,
NVIDIA_HDA_TRANSREG_ADDR ,
0x0f , NVIDIA_HDA_ENABLE_COHBITS ) ;
2008-08-20 16:43:24 -07:00
update_pci_byte ( chip - > pci ,
NVIDIA_HDA_ISTRM_COH ,
0x01 , NVIDIA_HDA_ENABLE_COHBIT ) ;
update_pci_byte ( chip - > pci ,
NVIDIA_HDA_OSTRM_COH ,
0x01 , NVIDIA_HDA_ENABLE_COHBIT ) ;
2011-05-25 09:11:37 +02:00
}
/* Enable SCH/PCH snoop if needed */
if ( chip - > driver_caps & AZX_DCAPS_SCH_SNOOP ) {
2011-09-28 17:16:09 +02:00
unsigned short snoop ;
2008-02-22 18:36:22 +01:00
pci_read_config_word ( chip - > pci , INTEL_SCH_HDA_DEVC , & snoop ) ;
2011-09-28 17:16:09 +02:00
if ( ( ! azx_snoop ( chip ) & & ! ( snoop & INTEL_SCH_HDA_DEVC_NOSNOOP ) ) | |
( azx_snoop ( chip ) & & ( snoop & INTEL_SCH_HDA_DEVC_NOSNOOP ) ) ) {
snoop & = ~ INTEL_SCH_HDA_DEVC_NOSNOOP ;
if ( ! azx_snoop ( chip ) )
snoop | = INTEL_SCH_HDA_DEVC_NOSNOOP ;
pci_write_config_word ( chip - > pci , INTEL_SCH_HDA_DEVC , snoop ) ;
2008-02-22 18:36:22 +01:00
pci_read_config_word ( chip - > pci ,
INTEL_SCH_HDA_DEVC , & snoop ) ;
}
2014-02-25 12:21:03 +01:00
dev_dbg ( chip - > card - > dev , " SCH snoop: %s \n " ,
( snoop & INTEL_SCH_HDA_DEVC_NOSNOOP ) ?
" Disabled " : " Enabled " ) ;
2005-09-13 18:49:12 +02:00
}
2005-04-16 15:20:36 -07:00
}
2014-06-26 16:50:16 +02:00
/* calculate runtime delay from LPIB */
static int azx_get_delay_from_lpib ( struct azx * chip , struct azx_dev * azx_dev ,
unsigned int pos )
{
struct snd_pcm_substream * substream = azx_dev - > substream ;
int stream = substream - > stream ;
unsigned int lpib_pos = azx_get_pos_lpib ( chip , azx_dev ) ;
int delay ;
if ( stream = = SNDRV_PCM_STREAM_PLAYBACK )
delay = pos - lpib_pos ;
else
delay = lpib_pos - pos ;
if ( delay < 0 ) {
if ( delay > = azx_dev - > delay_negative_threshold )
delay = 0 ;
else
delay + = azx_dev - > bufsize ;
}
if ( delay > = azx_dev - > period_bytes ) {
dev_info ( chip - > card - > dev ,
" Unstable LPIB (%d >= %d); disabling LPIB delay counting \n " ,
delay , azx_dev - > period_bytes ) ;
delay = 0 ;
chip - > driver_caps & = ~ AZX_DCAPS_COUNT_LPIB_DELAY ;
chip - > get_delay [ stream ] = NULL ;
}
return bytes_to_frames ( substream - > runtime , delay ) ;
}
2008-05-16 12:34:47 +02:00
static int azx_position_ok ( struct azx * chip , struct azx_dev * azx_dev ) ;
2014-02-28 15:41:28 -08:00
/* called from IRQ */
static int azx_position_check ( struct azx * chip , struct azx_dev * azx_dev )
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2014-02-28 15:41:28 -08:00
int ok ;
ok = azx_position_ok ( chip , azx_dev ) ;
if ( ok = = 1 ) {
azx_dev - > irq_pending = 0 ;
return ok ;
} else if ( ok = = 0 & & chip - > bus & & chip - > bus - > workq ) {
/* bogus IRQ, process it later */
azx_dev - > irq_pending = 1 ;
2014-06-26 17:19:20 +02:00
queue_work ( chip - > bus - > workq , & hda - > irq_pending_work ) ;
2014-02-28 15:41:28 -08:00
}
return 0 ;
}
2008-05-16 12:34:47 +02:00
/*
* Check whether the current DMA position is acceptable for updating
* periods . Returns non - zero if it ' s OK .
*
* Many HD - audio controllers appear pretty inaccurate about
* the update - IRQ timing . The IRQ is issued before actually the
* data is processed . So , we need to process it afterwords in a
* workqueue .
*/
static int azx_position_ok ( struct azx * chip , struct azx_dev * azx_dev )
{
2014-06-26 16:50:16 +02:00
struct snd_pcm_substream * substream = azx_dev - > substream ;
int stream = substream - > stream ;
2010-05-11 10:21:46 +02:00
u32 wallclk ;
2008-05-16 12:34:47 +02:00
unsigned int pos ;
2010-05-11 12:10:47 +02:00
wallclk = azx_readl ( chip , WALLCLK ) - azx_dev - > start_wallclk ;
if ( wallclk < ( azx_dev - > period_wallclk * 2 ) / 3 )
2009-04-10 12:20:45 +02:00
return - 1 ; /* bogus (too early) interrupt */
2014-06-26 16:50:16 +02:00
if ( chip - > get_position [ stream ] )
pos = chip - > get_position [ stream ] ( chip , azx_dev ) ;
else { /* use the position buffer as default */
pos = azx_get_pos_posbuf ( chip , azx_dev ) ;
if ( ! pos | | pos = = ( u32 ) - 1 ) {
dev_info ( chip - > card - > dev ,
" Invalid position buffer, using LPIB read method instead. \n " ) ;
chip - > get_position [ stream ] = azx_get_pos_lpib ;
pos = azx_get_pos_lpib ( chip , azx_dev ) ;
chip - > get_delay [ stream ] = NULL ;
} else {
chip - > get_position [ stream ] = azx_get_pos_posbuf ;
if ( chip - > driver_caps & AZX_DCAPS_COUNT_LPIB_DELAY )
chip - > get_delay [ stream ] = azx_get_delay_from_lpib ;
}
}
if ( pos > = azx_dev - > bufsize )
pos = 0 ;
2008-05-16 12:34:47 +02:00
2010-02-12 18:17:06 +01:00
if ( WARN_ONCE ( ! azx_dev - > period_bytes ,
" hda-intel: zero azx_dev->period_bytes " ) )
2010-05-11 12:10:47 +02:00
return - 1 ; /* this shouldn't happen! */
2010-06-02 13:29:17 +02:00
if ( wallclk < ( azx_dev - > period_wallclk * 5 ) / 4 & &
2010-05-11 12:10:47 +02:00
pos % azx_dev - > period_bytes > azx_dev - > period_bytes / 2 )
/* NG - it's below the first next period boundary */
2014-02-28 15:41:14 -08:00
return chip - > bdl_pos_adj [ chip - > dev_index ] ? 0 : - 1 ;
2010-06-02 13:29:17 +02:00
azx_dev - > start_wallclk + = wallclk ;
2008-05-16 12:34:47 +02:00
return 1 ; /* OK, it's fine */
}
/*
* The work for pending PCM period updates .
*/
static void azx_irq_pending_work ( struct work_struct * work )
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( work , struct hda_intel , irq_pending_work ) ;
struct azx * chip = & hda - > chip ;
2010-05-11 10:21:46 +02:00
int i , pending , ok ;
2008-05-16 12:34:47 +02:00
2014-06-26 17:19:20 +02:00
if ( ! hda - > irq_pending_warned ) {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev ,
" IRQ timing workaround is activated for card #%d. Suggest a bigger bdl_pos_adj. \n " ,
chip - > card - > number ) ;
2014-06-26 17:19:20 +02:00
hda - > irq_pending_warned = 1 ;
2008-06-10 17:53:35 +02:00
}
2008-05-16 12:34:47 +02:00
for ( ; ; ) {
pending = 0 ;
spin_lock_irq ( & chip - > reg_lock ) ;
for ( i = 0 ; i < chip - > num_streams ; i + + ) {
struct azx_dev * azx_dev = & chip - > azx_dev [ i ] ;
if ( ! azx_dev - > irq_pending | |
! azx_dev - > substream | |
! azx_dev - > running )
continue ;
2010-05-11 10:21:46 +02:00
ok = azx_position_ok ( chip , azx_dev ) ;
if ( ok > 0 ) {
2008-05-16 12:34:47 +02:00
azx_dev - > irq_pending = 0 ;
spin_unlock ( & chip - > reg_lock ) ;
snd_pcm_period_elapsed ( azx_dev - > substream ) ;
spin_lock ( & chip - > reg_lock ) ;
2010-05-11 10:21:46 +02:00
} else if ( ok < 0 ) {
pending = 0 ; /* too early */
2008-05-16 12:34:47 +02:00
} else
pending + + ;
}
spin_unlock_irq ( & chip - > reg_lock ) ;
if ( ! pending )
return ;
2010-08-03 14:39:04 +02:00
msleep ( 1 ) ;
2008-05-16 12:34:47 +02:00
}
}
/* clear irq_pending flags and assure no on-going workq */
static void azx_clear_irq_pending ( struct azx * chip )
{
int i ;
spin_lock_irq ( & chip - > reg_lock ) ;
for ( i = 0 ; i < chip - > num_streams ; i + + )
chip - > azx_dev [ i ] . irq_pending = 0 ;
spin_unlock_irq ( & chip - > reg_lock ) ;
2005-04-16 15:20:36 -07:00
}
2006-10-23 13:40:59 +02:00
static int azx_acquire_irq ( struct azx * chip , int do_disconnect )
{
2006-11-21 12:14:23 +01:00
if ( request_irq ( chip - > pci - > irq , azx_interrupt ,
chip - > msi ? 0 : IRQF_SHARED ,
2011-06-10 16:36:37 +02:00
KBUILD_MODNAME , chip ) ) {
2014-02-25 12:21:03 +01:00
dev_err ( chip - > card - > dev ,
" unable to grab IRQ %d, disabling device \n " ,
chip - > pci - > irq ) ;
2006-10-23 13:40:59 +02:00
if ( do_disconnect )
snd_card_disconnect ( chip - > card ) ;
return - 1 ;
}
chip - > irq = chip - > pci - > irq ;
2006-11-21 12:10:55 +01:00
pci_intx ( chip - > pci , ! chip - > msi ) ;
2006-10-23 13:40:59 +02:00
return 0 ;
}
2014-06-26 16:50:16 +02:00
/* get the current DMA position with correction on VIA chips */
static unsigned int azx_via_get_position ( struct azx * chip ,
struct azx_dev * azx_dev )
{
unsigned int link_pos , mini_pos , bound_pos ;
unsigned int mod_link_pos , mod_dma_pos , mod_mini_pos ;
unsigned int fifo_size ;
link_pos = azx_sd_readl ( chip , azx_dev , SD_LPIB ) ;
if ( azx_dev - > substream - > stream = = SNDRV_PCM_STREAM_PLAYBACK ) {
/* Playback, no problem using link position */
return link_pos ;
}
/* Capture */
/* For new chipset,
* use mod to get the DMA position just like old chipset
*/
mod_dma_pos = le32_to_cpu ( * azx_dev - > posbuf ) ;
mod_dma_pos % = azx_dev - > period_bytes ;
/* azx_dev->fifo_size can't get FIFO size of in stream.
* Get from base address + offset .
*/
fifo_size = readw ( chip - > remap_addr + VIA_IN_STREAM0_FIFO_SIZE_OFFSET ) ;
if ( azx_dev - > insufficient ) {
/* Link position never gather than FIFO size */
if ( link_pos < = fifo_size )
return 0 ;
azx_dev - > insufficient = 0 ;
}
if ( link_pos < = fifo_size )
mini_pos = azx_dev - > bufsize + link_pos - fifo_size ;
else
mini_pos = link_pos - fifo_size ;
/* Find nearest previous boudary */
mod_mini_pos = mini_pos % azx_dev - > period_bytes ;
mod_link_pos = link_pos % azx_dev - > period_bytes ;
if ( mod_link_pos > = fifo_size )
bound_pos = link_pos - mod_link_pos ;
else if ( mod_dma_pos > = mod_mini_pos )
bound_pos = mini_pos - mod_mini_pos ;
else {
bound_pos = mini_pos - mod_mini_pos + azx_dev - > period_bytes ;
if ( bound_pos > = azx_dev - > bufsize )
bound_pos = 0 ;
}
/* Calculate real DMA position we want */
return bound_pos + mod_dma_pos ;
}
2012-08-24 18:38:08 +02:00
# ifdef CONFIG_PM
2012-08-14 17:13:32 +02:00
static DEFINE_MUTEX ( card_list_lock ) ;
static LIST_HEAD ( card_list ) ;
static void azx_add_card_list ( struct azx * chip )
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2012-08-14 17:13:32 +02:00
mutex_lock ( & card_list_lock ) ;
2014-06-26 17:19:20 +02:00
list_add ( & hda - > list , & card_list ) ;
2012-08-14 17:13:32 +02:00
mutex_unlock ( & card_list_lock ) ;
}
static void azx_del_card_list ( struct azx * chip )
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2012-08-14 17:13:32 +02:00
mutex_lock ( & card_list_lock ) ;
2014-06-26 17:19:20 +02:00
list_del_init ( & hda - > list ) ;
2012-08-14 17:13:32 +02:00
mutex_unlock ( & card_list_lock ) ;
}
/* trigger power-save check at writing parameter */
static int param_set_xint ( const char * val , const struct kernel_param * kp )
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda ;
2012-08-14 17:13:32 +02:00
struct azx * chip ;
struct hda_codec * c ;
int prev = power_save ;
int ret = param_set_int ( val , kp ) ;
if ( ret | | prev = = power_save )
return ret ;
mutex_lock ( & card_list_lock ) ;
2014-06-26 17:19:20 +02:00
list_for_each_entry ( hda , & card_list , list ) {
chip = & hda - > chip ;
2012-08-14 17:13:32 +02:00
if ( ! chip - > bus | | chip - > disabled )
continue ;
list_for_each_entry ( c , & chip - > bus - > codec_list , list )
snd_hda_power_sync ( c ) ;
}
mutex_unlock ( & card_list_lock ) ;
return 0 ;
}
# else
# define azx_add_card_list(chip) /* NOP */
# define azx_del_card_list(chip) /* NOP */
2012-08-24 18:38:08 +02:00
# endif /* CONFIG_PM */
2008-12-11 11:47:17 +01:00
2012-08-14 18:10:09 +02:00
# if defined(CONFIG_PM_SLEEP) || defined(SUPPORT_VGA_SWITCHEROO)
2008-12-11 11:47:17 +01:00
/*
* power management
*/
2012-07-02 15:20:37 +02:00
static int azx_suspend ( struct device * dev )
2005-04-16 15:20:36 -07:00
{
2012-07-02 15:20:37 +02:00
struct pci_dev * pci = to_pci_dev ( dev ) ;
struct snd_card * card = dev_get_drvdata ( dev ) ;
2014-07-16 16:31:04 +02:00
struct azx * chip ;
struct hda_intel * hda ;
2011-11-24 14:31:46 +01:00
struct azx_pcm * p ;
2005-04-16 15:20:36 -07:00
2014-07-16 16:31:04 +02:00
if ( ! card )
return 0 ;
chip = card - > private_data ;
hda = container_of ( chip , struct hda_intel , chip ) ;
2014-07-15 15:27:19 +02:00
if ( chip - > disabled | | hda - > init_failed )
2012-12-04 17:01:25 +01:00
return 0 ;
2005-11-17 16:11:09 +01:00
snd_power_change_state ( card , SNDRV_CTL_POWER_D3hot ) ;
2008-05-16 12:34:47 +02:00
azx_clear_irq_pending ( chip ) ;
2011-11-24 14:31:46 +01:00
list_for_each_entry ( p , & chip - > pcm_list , list )
snd_pcm_suspend_all ( p - > pcm ) ;
2007-08-14 15:18:26 +02:00
if ( chip - > initialized )
2009-06-02 01:16:07 +02:00
snd_hda_suspend ( chip - > bus ) ;
2007-08-10 17:21:45 +02:00
azx_stop_chip ( chip ) ;
2013-06-25 05:58:49 -04:00
azx_enter_link_reset ( chip ) ;
2006-10-11 18:52:53 +02:00
if ( chip - > irq > = 0 ) {
2006-09-08 12:30:03 +02:00
free_irq ( chip - > irq , chip ) ;
2006-10-11 18:52:53 +02:00
chip - > irq = - 1 ;
}
2014-06-26 18:45:16 +08:00
2006-10-23 13:40:59 +02:00
if ( chip - > msi )
2006-09-08 12:30:03 +02:00
pci_disable_msi ( chip - > pci ) ;
2005-11-17 16:11:09 +01:00
pci_disable_device ( pci ) ;
pci_save_state ( pci ) ;
2012-07-02 15:20:37 +02:00
pci_set_power_state ( pci , PCI_D3hot ) ;
2013-05-30 22:07:10 +08:00
if ( chip - > driver_caps & AZX_DCAPS_I915_POWERWELL )
hda_display_power ( false ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2012-07-02 15:20:37 +02:00
static int azx_resume ( struct device * dev )
2005-04-16 15:20:36 -07:00
{
2012-07-02 15:20:37 +02:00
struct pci_dev * pci = to_pci_dev ( dev ) ;
struct snd_card * card = dev_get_drvdata ( dev ) ;
2014-07-16 16:31:04 +02:00
struct azx * chip ;
struct hda_intel * hda ;
if ( ! card )
return 0 ;
2005-04-16 15:20:36 -07:00
2014-07-16 16:31:04 +02:00
chip = card - > private_data ;
hda = container_of ( chip , struct hda_intel , chip ) ;
2014-07-15 15:27:19 +02:00
if ( chip - > disabled | | hda - > init_failed )
2012-12-04 17:01:25 +01:00
return 0 ;
2014-06-26 18:45:16 +08:00
if ( chip - > driver_caps & AZX_DCAPS_I915_POWERWELL ) {
2013-05-30 22:07:10 +08:00
hda_display_power ( true ) ;
2014-07-03 17:02:23 +08:00
haswell_set_bclk ( chip ) ;
2014-06-26 18:45:16 +08:00
}
2009-02-16 10:13:03 +01:00
pci_set_power_state ( pci , PCI_D0 ) ;
pci_restore_state ( pci ) ;
2006-10-11 18:52:53 +02:00
if ( pci_enable_device ( pci ) < 0 ) {
2014-02-25 12:21:03 +01:00
dev_err ( chip - > card - > dev ,
" pci_enable_device failed, disabling device \n " ) ;
2006-10-11 18:52:53 +02:00
snd_card_disconnect ( card ) ;
return - EIO ;
}
pci_set_master ( pci ) ;
2006-10-23 13:40:59 +02:00
if ( chip - > msi )
if ( pci_enable_msi ( pci ) < 0 )
chip - > msi = 0 ;
if ( azx_acquire_irq ( chip , 1 ) < 0 )
2006-10-11 18:52:53 +02:00
return - EIO ;
2007-08-10 17:21:45 +02:00
azx_init_pci ( chip ) ;
2007-09-03 15:28:04 +02:00
2014-04-09 12:30:57 +02:00
azx_init_chip ( chip , true ) ;
2007-09-03 15:28:04 +02:00
2005-04-16 15:20:36 -07:00
snd_hda_resume ( chip - > bus ) ;
2005-11-17 16:11:09 +01:00
snd_power_change_state ( card , SNDRV_CTL_POWER_D0 ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2012-08-23 17:32:30 +08:00
# endif /* CONFIG_PM_SLEEP || SUPPORT_VGA_SWITCHEROO */
# ifdef CONFIG_PM_RUNTIME
static int azx_runtime_suspend ( struct device * dev )
{
struct snd_card * card = dev_get_drvdata ( dev ) ;
2014-07-16 16:31:04 +02:00
struct azx * chip ;
struct hda_intel * hda ;
2012-08-23 17:32:30 +08:00
2014-07-16 16:31:04 +02:00
if ( ! card )
return 0 ;
chip = card - > private_data ;
hda = container_of ( chip , struct hda_intel , chip ) ;
2014-07-15 15:27:19 +02:00
if ( chip - > disabled | | hda - > init_failed )
2013-07-29 15:19:29 +10:00
return 0 ;
if ( ! ( chip - > driver_caps & AZX_DCAPS_PM_RUNTIME ) )
return 0 ;
2013-07-25 23:34:46 -04:00
/* enable controller wake up event */
azx_writew ( chip , WAKEEN , azx_readw ( chip , WAKEEN ) |
STATESTS_INT_MASK ) ;
2012-08-23 17:32:30 +08:00
azx_stop_chip ( chip ) ;
2013-11-26 11:58:40 +01:00
azx_enter_link_reset ( chip ) ;
2012-08-23 17:32:30 +08:00
azx_clear_irq_pending ( chip ) ;
2014-07-03 17:02:23 +08:00
if ( chip - > driver_caps & AZX_DCAPS_I915_POWERWELL )
2013-05-30 22:07:10 +08:00
hda_display_power ( false ) ;
2014-07-03 17:02:23 +08:00
2012-08-23 17:32:30 +08:00
return 0 ;
}
static int azx_runtime_resume ( struct device * dev )
{
struct snd_card * card = dev_get_drvdata ( dev ) ;
2014-07-16 16:31:04 +02:00
struct azx * chip ;
struct hda_intel * hda ;
2013-07-25 23:34:46 -04:00
struct hda_bus * bus ;
struct hda_codec * codec ;
int status ;
2012-08-23 17:32:30 +08:00
2014-07-16 16:31:04 +02:00
if ( ! card )
return 0 ;
chip = card - > private_data ;
hda = container_of ( chip , struct hda_intel , chip ) ;
2014-07-15 15:27:19 +02:00
if ( chip - > disabled | | hda - > init_failed )
2013-07-29 15:19:29 +10:00
return 0 ;
if ( ! ( chip - > driver_caps & AZX_DCAPS_PM_RUNTIME ) )
return 0 ;
2014-06-26 18:45:16 +08:00
if ( chip - > driver_caps & AZX_DCAPS_I915_POWERWELL ) {
2013-05-30 22:07:10 +08:00
hda_display_power ( true ) ;
2014-07-03 17:02:23 +08:00
haswell_set_bclk ( chip ) ;
2014-06-26 18:45:16 +08:00
}
2013-07-25 23:34:46 -04:00
/* Read STATESTS before controller reset */
status = azx_readw ( chip , STATESTS ) ;
2012-08-23 17:32:30 +08:00
azx_init_pci ( chip ) ;
2014-04-09 12:30:57 +02:00
azx_init_chip ( chip , true ) ;
2013-07-25 23:34:46 -04:00
bus = chip - > bus ;
if ( status & & bus ) {
list_for_each_entry ( codec , & bus - > codec_list , list )
if ( status & ( 1 < < codec - > addr ) )
queue_delayed_work ( codec - > bus - > workq ,
& codec - > jackpoll_work , codec - > jackpoll_interval ) ;
}
/* disable controller Wake Up event*/
azx_writew ( chip , WAKEEN , azx_readw ( chip , WAKEEN ) &
~ STATESTS_INT_MASK ) ;
2012-08-23 17:32:30 +08:00
return 0 ;
}
2012-12-12 11:50:12 +01:00
static int azx_runtime_idle ( struct device * dev )
{
struct snd_card * card = dev_get_drvdata ( dev ) ;
2014-07-16 16:31:04 +02:00
struct azx * chip ;
struct hda_intel * hda ;
if ( ! card )
return 0 ;
2012-12-12 11:50:12 +01:00
2014-07-16 16:31:04 +02:00
chip = card - > private_data ;
hda = container_of ( chip , struct hda_intel , chip ) ;
2014-07-15 15:27:19 +02:00
if ( chip - > disabled | | hda - > init_failed )
2013-07-29 15:19:29 +10:00
return 0 ;
2012-12-12 11:50:12 +01:00
if ( ! power_save_controller | |
! ( chip - > driver_caps & AZX_DCAPS_PM_RUNTIME ) )
return - EBUSY ;
return 0 ;
}
2012-08-23 17:32:30 +08:00
# endif /* CONFIG_PM_RUNTIME */
# ifdef CONFIG_PM
static const struct dev_pm_ops azx_pm = {
SET_SYSTEM_SLEEP_PM_OPS ( azx_suspend , azx_resume )
2012-12-12 11:50:12 +01:00
SET_RUNTIME_PM_OPS ( azx_runtime_suspend , azx_runtime_resume , azx_runtime_idle )
2012-08-23 17:32:30 +08:00
} ;
2012-07-02 15:20:37 +02:00
# define AZX_PM_OPS &azx_pm
# else
# define AZX_PM_OPS NULL
2012-08-23 17:32:30 +08:00
# endif /* CONFIG_PM */
2005-04-16 15:20:36 -07:00
2012-12-07 07:40:35 +01:00
static int azx_probe_continue ( struct azx * chip ) ;
2012-04-26 12:23:42 +02:00
2012-06-08 13:06:29 +02:00
# ifdef SUPPORT_VGA_SWITCHEROO
2012-12-06 12:35:10 -05:00
static struct pci_dev * get_bound_vga ( struct pci_dev * pci ) ;
2012-04-26 12:23:42 +02:00
static void azx_vs_set_state ( struct pci_dev * pci ,
enum vga_switcheroo_state state )
{
struct snd_card * card = pci_get_drvdata ( pci ) ;
struct azx * chip = card - > private_data ;
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2012-04-26 12:23:42 +02:00
bool disabled ;
2014-06-26 17:19:20 +02:00
wait_for_completion ( & hda - > probe_wait ) ;
if ( hda - > init_failed )
2012-04-26 12:23:42 +02:00
return ;
disabled = ( state = = VGA_SWITCHEROO_OFF ) ;
if ( chip - > disabled = = disabled )
return ;
if ( ! chip - > bus ) {
chip - > disabled = disabled ;
if ( ! disabled ) {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev ,
" Start delayed initialization \n " ) ;
2013-05-30 22:07:09 +08:00
if ( azx_probe_continue ( chip ) < 0 ) {
2014-02-25 12:21:03 +01:00
dev_err ( chip - > card - > dev , " initialization error \n " ) ;
2014-06-26 17:19:20 +02:00
hda - > init_failed = true ;
2012-04-26 12:23:42 +02:00
}
}
} else {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev , " %s via VGA-switcheroo \n " ,
disabled ? " Disabling " : " Enabling " ) ;
2012-04-26 12:23:42 +02:00
if ( disabled ) {
2014-02-28 15:41:15 -08:00
pm_runtime_put_sync_suspend ( card - > dev ) ;
azx_suspend ( card - > dev ) ;
2013-07-29 15:19:29 +10:00
/* when we get suspended by vga switcheroo we end up in D3cold,
* however we have no ACPI handle , so pci / acpi can ' t put us there ,
* put ourselves there */
pci - > current_state = PCI_D3cold ;
2012-04-26 12:23:42 +02:00
chip - > disabled = true ;
2012-10-12 17:28:18 +02:00
if ( snd_hda_lock_devices ( chip - > bus ) )
2014-02-25 12:21:03 +01:00
dev_warn ( chip - > card - > dev ,
" Cannot lock devices! \n " ) ;
2012-04-26 12:23:42 +02:00
} else {
snd_hda_unlock_devices ( chip - > bus ) ;
2014-02-28 15:41:15 -08:00
pm_runtime_get_noresume ( card - > dev ) ;
2012-04-26 12:23:42 +02:00
chip - > disabled = false ;
2014-02-28 15:41:15 -08:00
azx_resume ( card - > dev ) ;
2012-04-26 12:23:42 +02:00
}
}
}
static bool azx_vs_can_switch ( struct pci_dev * pci )
{
struct snd_card * card = pci_get_drvdata ( pci ) ;
struct azx * chip = card - > private_data ;
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2012-04-26 12:23:42 +02:00
2014-06-26 17:19:20 +02:00
wait_for_completion ( & hda - > probe_wait ) ;
if ( hda - > init_failed )
2012-04-26 12:23:42 +02:00
return false ;
if ( chip - > disabled | | ! chip - > bus )
return true ;
if ( snd_hda_lock_devices ( chip - > bus ) )
return false ;
snd_hda_unlock_devices ( chip - > bus ) ;
return true ;
}
2012-12-06 12:35:10 -05:00
static void init_vga_switcheroo ( struct azx * chip )
2012-04-26 12:23:42 +02:00
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2012-04-26 12:23:42 +02:00
struct pci_dev * p = get_bound_vga ( chip - > pci ) ;
if ( p ) {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev ,
" Handle VGA-switcheroo audio client \n " ) ;
2014-06-26 17:19:20 +02:00
hda - > use_vga_switcheroo = 1 ;
2012-04-26 12:23:42 +02:00
pci_dev_put ( p ) ;
}
}
static const struct vga_switcheroo_client_ops azx_vs_ops = {
. set_gpu_state = azx_vs_set_state ,
. can_switch = azx_vs_can_switch ,
} ;
2012-12-06 12:35:10 -05:00
static int register_vga_switcheroo ( struct azx * chip )
2012-04-26 12:23:42 +02:00
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2012-10-12 17:28:18 +02:00
int err ;
2014-06-26 17:19:20 +02:00
if ( ! hda - > use_vga_switcheroo )
2012-04-26 12:23:42 +02:00
return 0 ;
/* FIXME: currently only handling DIS controller
* is there any machine with two switchable HDMI audio controllers ?
*/
2012-10-12 17:28:18 +02:00
err = vga_switcheroo_register_audio_client ( chip - > pci , & azx_vs_ops ,
2012-04-26 12:23:42 +02:00
VGA_SWITCHEROO_DIS ,
chip - > bus ! = NULL ) ;
2012-10-12 17:28:18 +02:00
if ( err < 0 )
return err ;
2014-06-26 17:19:20 +02:00
hda - > vga_switcheroo_registered = 1 ;
2013-07-29 15:19:29 +10:00
/* register as an optimus hdmi audio power domain */
2014-02-28 15:41:15 -08:00
vga_switcheroo_init_domain_pm_optimus_hdmi_audio ( chip - > card - > dev ,
2014-06-26 17:19:20 +02:00
& hda - > hdmi_pm_domain ) ;
2012-10-12 17:28:18 +02:00
return 0 ;
2012-04-26 12:23:42 +02:00
}
# else
# define init_vga_switcheroo(chip) /* NOP */
# define register_vga_switcheroo(chip) 0
2012-06-08 13:06:29 +02:00
# define check_hdmi_disabled(pci) false
2012-04-26 12:23:42 +02:00
# endif /* SUPPORT_VGA_SWITCHER */
2005-04-16 15:20:36 -07:00
/*
* destructor
*/
2005-11-17 14:59:02 +01:00
static int azx_free ( struct azx * chip )
2005-04-16 15:20:36 -07:00
{
2013-05-30 22:07:08 +08:00
struct pci_dev * pci = chip - > pci ;
2014-06-26 18:45:16 +08:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2008-02-06 14:50:19 +01:00
int i ;
2013-05-30 22:07:08 +08:00
if ( ( chip - > driver_caps & AZX_DCAPS_PM_RUNTIME )
& & chip - > running )
pm_runtime_get_noresume ( & pci - > dev ) ;
2012-08-14 17:13:32 +02:00
azx_del_card_list ( chip ) ;
2008-10-29 16:18:25 +01:00
azx_notifier_unregister ( chip ) ;
2014-06-26 17:19:20 +02:00
hda - > init_failed = 1 ; /* to be sure */
complete_all ( & hda - > probe_wait ) ;
2012-12-04 15:09:23 +01:00
2014-06-26 17:19:20 +02:00
if ( use_vga_switcheroo ( hda ) ) {
2012-04-26 12:23:42 +02:00
if ( chip - > disabled & & chip - > bus )
snd_hda_unlock_devices ( chip - > bus ) ;
2014-06-26 17:19:20 +02:00
if ( hda - > vga_switcheroo_registered )
2012-10-12 17:28:18 +02:00
vga_switcheroo_unregister_client ( chip - > pci ) ;
2012-04-26 12:23:42 +02:00
}
2005-05-30 20:33:44 +02:00
if ( chip - > initialized ) {
2008-05-16 12:34:47 +02:00
azx_clear_irq_pending ( chip ) ;
2005-08-24 14:14:57 +02:00
for ( i = 0 ; i < chip - > num_streams ; i + + )
2005-04-16 15:20:36 -07:00
azx_stream_stop ( chip , & chip - > azx_dev [ i ] ) ;
2007-08-10 17:21:45 +02:00
azx_stop_chip ( chip ) ;
2005-04-16 15:20:36 -07:00
}
2008-04-22 13:50:34 +02:00
if ( chip - > irq > = 0 )
2005-04-16 15:20:36 -07:00
free_irq ( chip - > irq , ( void * ) chip ) ;
2006-10-23 13:40:59 +02:00
if ( chip - > msi )
2006-10-11 18:52:53 +02:00
pci_disable_msi ( chip - > pci ) ;
2006-06-01 11:42:14 +02:00
if ( chip - > remap_addr )
iounmap ( chip - > remap_addr ) ;
2005-04-16 15:20:36 -07:00
2014-02-28 15:41:23 -08:00
azx_free_stream_pages ( chip ) ;
2012-04-26 12:23:42 +02:00
if ( chip - > region_requested )
pci_release_regions ( chip - > pci ) ;
2005-04-16 15:20:36 -07:00
pci_disable_device ( chip - > pci ) ;
2005-08-24 14:14:57 +02:00
kfree ( chip - > azx_dev ) ;
2012-08-09 12:33:28 +02:00
# ifdef CONFIG_SND_HDA_PATCH_LOADER
if ( chip - > fw )
release_firmware ( chip - > fw ) ;
# endif
2013-05-30 22:07:10 +08:00
if ( chip - > driver_caps & AZX_DCAPS_I915_POWERWELL ) {
hda_display_power ( false ) ;
hda_i915_exit ( ) ;
}
2014-06-26 18:45:16 +08:00
kfree ( hda ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2005-11-17 14:59:02 +01:00
static int azx_dev_free ( struct snd_device * device )
2005-04-16 15:20:36 -07:00
{
return azx_free ( device - > device_data ) ;
}
2012-06-08 13:06:29 +02:00
# ifdef SUPPORT_VGA_SWITCHEROO
2012-04-26 12:13:25 +02:00
/*
* Check of disabled HDMI controller by vga - switcheroo
*/
2012-12-06 12:35:10 -05:00
static struct pci_dev * get_bound_vga ( struct pci_dev * pci )
2012-04-26 12:13:25 +02:00
{
struct pci_dev * p ;
/* check only discrete GPU */
switch ( pci - > vendor ) {
case PCI_VENDOR_ID_ATI :
case PCI_VENDOR_ID_AMD :
case PCI_VENDOR_ID_NVIDIA :
if ( pci - > devfn = = 1 ) {
p = pci_get_domain_bus_and_slot ( pci_domain_nr ( pci - > bus ) ,
pci - > bus - > number , 0 ) ;
if ( p ) {
if ( ( p - > class > > 8 ) = = PCI_CLASS_DISPLAY_VGA )
return p ;
pci_dev_put ( p ) ;
}
}
break ;
}
return NULL ;
}
2012-12-06 12:35:10 -05:00
static bool check_hdmi_disabled ( struct pci_dev * pci )
2012-04-26 12:13:25 +02:00
{
bool vga_inactive = false ;
struct pci_dev * p = get_bound_vga ( pci ) ;
if ( p ) {
2012-06-07 12:15:16 +02:00
if ( vga_switcheroo_get_client_state ( p ) = = VGA_SWITCHEROO_OFF )
2012-04-26 12:13:25 +02:00
vga_inactive = true ;
pci_dev_put ( p ) ;
}
return vga_inactive ;
}
2012-06-08 13:06:29 +02:00
# endif /* SUPPORT_VGA_SWITCHEROO */
2012-04-26 12:13:25 +02:00
2007-02-01 15:46:50 +01:00
/*
* white / black - listing for position_fix
*/
2012-12-06 12:35:10 -05:00
static struct snd_pci_quirk position_fix_list [ ] = {
2008-06-10 17:53:34 +02:00
SND_PCI_QUIRK ( 0x1028 , 0x01cc , " Dell D820 " , POS_FIX_LPIB ) ,
SND_PCI_QUIRK ( 0x1028 , 0x01de , " Dell Precision 390 " , POS_FIX_LPIB ) ,
2009-12-01 14:17:37 +01:00
SND_PCI_QUIRK ( 0x103c , 0x306d , " HP dv3 " , POS_FIX_LPIB ) ,
2008-06-10 17:53:34 +02:00
SND_PCI_QUIRK ( 0x1043 , 0x813d , " ASUS P5AD2 " , POS_FIX_LPIB ) ,
2010-05-30 01:17:03 -04:00
SND_PCI_QUIRK ( 0x1043 , 0x81b3 , " ASUS " , POS_FIX_LPIB ) ,
2010-05-30 13:08:41 -04:00
SND_PCI_QUIRK ( 0x1043 , 0x81e7 , " ASUS M2V " , POS_FIX_LPIB ) ,
2010-05-27 18:32:18 -04:00
SND_PCI_QUIRK ( 0x104d , 0x9069 , " Sony VPCS11V9E " , POS_FIX_LPIB ) ,
2012-01-12 16:31:14 +01:00
SND_PCI_QUIRK ( 0x10de , 0xcb89 , " Macbook Pro 7,1 " , POS_FIX_LPIB ) ,
2010-05-29 11:04:11 -04:00
SND_PCI_QUIRK ( 0x1297 , 0x3166 , " Shuttle " , POS_FIX_LPIB ) ,
2010-03-28 02:34:40 -04:00
SND_PCI_QUIRK ( 0x1458 , 0xa022 , " ga-ma770-ud3 " , POS_FIX_LPIB ) ,
2009-11-30 11:58:30 +01:00
SND_PCI_QUIRK ( 0x1462 , 0x1002 , " MSI Wind U115 " , POS_FIX_LPIB ) ,
2010-04-15 09:02:41 +02:00
SND_PCI_QUIRK ( 0x1565 , 0x8218 , " Biostar Microtech " , POS_FIX_LPIB ) ,
2010-05-30 19:31:41 -04:00
SND_PCI_QUIRK ( 0x1849 , 0x0888 , " 775Dual-VSTA " , POS_FIX_LPIB ) ,
2010-04-21 19:55:43 -04:00
SND_PCI_QUIRK ( 0x8086 , 0x2503 , " DG965OT AAD63733-203 " , POS_FIX_LPIB ) ,
2007-02-01 15:46:50 +01:00
{ }
} ;
2012-12-06 12:35:10 -05:00
static int check_position_fix ( struct azx * chip , int fix )
2007-02-01 15:46:50 +01:00
{
const struct snd_pci_quirk * q ;
2009-03-17 07:49:14 +01:00
switch ( fix ) {
2012-09-13 14:59:47 +02:00
case POS_FIX_AUTO :
2009-03-17 07:49:14 +01:00
case POS_FIX_LPIB :
case POS_FIX_POSBUF :
2010-09-30 10:12:50 +02:00
case POS_FIX_VIACOMBO :
2012-02-28 11:58:40 +01:00
case POS_FIX_COMBO :
2009-03-17 07:49:14 +01:00
return fix ;
}
q = snd_pci_quirk_lookup ( chip - > pci , position_fix_list ) ;
if ( q ) {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev ,
" position_fix set to %d for device %04x:%04x \n " ,
q - > value , q - > subvendor , q - > subdevice ) ;
2009-03-17 07:49:14 +01:00
return q - > value ;
2007-02-01 15:46:50 +01:00
}
2010-10-04 12:02:14 +02:00
/* Check VIA/ATI HD Audio Controller exist */
2011-05-25 09:11:37 +02:00
if ( chip - > driver_caps & AZX_DCAPS_POSFIX_VIA ) {
2014-02-25 12:21:03 +01:00
dev_dbg ( chip - > card - > dev , " Using VIACOMBO position fix \n " ) ;
2010-10-04 12:02:14 +02:00
return POS_FIX_VIACOMBO ;
2011-05-25 09:11:37 +02:00
}
if ( chip - > driver_caps & AZX_DCAPS_POSFIX_LPIB ) {
2014-02-25 12:21:03 +01:00
dev_dbg ( chip - > card - > dev , " Using LPIB position fix \n " ) ;
2011-05-20 16:29:09 +02:00
return POS_FIX_LPIB ;
2010-10-04 12:02:14 +02:00
}
2009-03-17 07:49:14 +01:00
return POS_FIX_AUTO ;
2007-02-01 15:46:50 +01:00
}
2014-06-26 16:50:16 +02:00
static void assign_position_fix ( struct azx * chip , int fix )
{
static azx_get_pos_callback_t callbacks [ ] = {
[ POS_FIX_AUTO ] = NULL ,
[ POS_FIX_LPIB ] = azx_get_pos_lpib ,
[ POS_FIX_POSBUF ] = azx_get_pos_posbuf ,
[ POS_FIX_VIACOMBO ] = azx_via_get_position ,
[ POS_FIX_COMBO ] = azx_get_pos_lpib ,
} ;
chip - > get_position [ 0 ] = chip - > get_position [ 1 ] = callbacks [ fix ] ;
/* combo mode uses LPIB only for playback */
if ( fix = = POS_FIX_COMBO )
chip - > get_position [ 1 ] = NULL ;
if ( fix = = POS_FIX_POSBUF & &
( chip - > driver_caps & AZX_DCAPS_COUNT_LPIB_DELAY ) ) {
chip - > get_delay [ 0 ] = chip - > get_delay [ 1 ] =
azx_get_delay_from_lpib ;
}
}
2007-08-17 09:17:36 +02:00
/*
* black - lists for probe_mask
*/
2012-12-06 12:35:10 -05:00
static struct snd_pci_quirk probe_mask_list [ ] = {
2007-08-17 09:17:36 +02:00
/* Thinkpad often breaks the controller communication when accessing
* to the non - working ( or non - existing ) modem codec slot .
*/
SND_PCI_QUIRK ( 0x1014 , 0x05b7 , " Thinkpad Z60 " , 0x01 ) ,
SND_PCI_QUIRK ( 0x17aa , 0x2010 , " Thinkpad X/T/R60 " , 0x01 ) ,
SND_PCI_QUIRK ( 0x17aa , 0x20ac , " Thinkpad X/T/R61 " , 0x01 ) ,
2008-11-07 14:53:09 +01:00
/* broken BIOS */
SND_PCI_QUIRK ( 0x1028 , 0x20ac , " Dell Studio Desktop " , 0x01 ) ,
2008-11-24 17:29:28 +01:00
/* including bogus ALC268 in slot#2 that conflicts with ALC888 */
SND_PCI_QUIRK ( 0x17c0 , 0x4085 , " Medion MD96630 " , 0x01 ) ,
2009-02-13 08:18:48 +01:00
/* forced codec slots */
2009-05-23 15:00:04 +03:00
SND_PCI_QUIRK ( 0x1043 , 0x1262 , " ASUS W5Fm " , 0x103 ) ,
2009-02-13 08:18:48 +01:00
SND_PCI_QUIRK ( 0x1046 , 0x1262 , " ASUS W5F " , 0x103 ) ,
2012-04-26 17:52:35 +02:00
/* WinFast VP200 H (Teradici) user reported broken communication */
SND_PCI_QUIRK ( 0x3a21 , 0x040d , " WinFast VP200 H " , 0x101 ) ,
2007-08-17 09:17:36 +02:00
{ }
} ;
2009-02-13 08:16:55 +01:00
# define AZX_FORCE_CODEC_MASK 0x100
2012-12-06 12:35:10 -05:00
static void check_probe_mask ( struct azx * chip , int dev )
2007-08-17 09:17:36 +02:00
{
const struct snd_pci_quirk * q ;
2009-02-13 08:16:55 +01:00
chip - > codec_probe_mask = probe_mask [ dev ] ;
if ( chip - > codec_probe_mask = = - 1 ) {
2007-08-17 09:17:36 +02:00
q = snd_pci_quirk_lookup ( chip - > pci , probe_mask_list ) ;
if ( q ) {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev ,
" probe_mask set to 0x%x for device %04x:%04x \n " ,
q - > value , q - > subvendor , q - > subdevice ) ;
2009-02-13 08:16:55 +01:00
chip - > codec_probe_mask = q - > value ;
2007-08-17 09:17:36 +02:00
}
}
2009-02-13 08:16:55 +01:00
/* check forced option */
if ( chip - > codec_probe_mask ! = - 1 & &
( chip - > codec_probe_mask & AZX_FORCE_CODEC_MASK ) ) {
chip - > codec_mask = chip - > codec_probe_mask & 0xff ;
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev , " codec_mask forced to 0x%x \n " ,
chip - > codec_mask ) ;
2009-02-13 08:16:55 +01:00
}
2007-08-17 09:17:36 +02:00
}
2009-08-11 14:21:26 +02:00
/*
2009-09-28 13:14:04 +02:00
* white / black - list for enable_msi
2009-08-11 14:21:26 +02:00
*/
2012-12-06 12:35:10 -05:00
static struct snd_pci_quirk msi_black_list [ ] = {
2013-12-12 09:52:03 +01:00
SND_PCI_QUIRK ( 0x103c , 0x2191 , " HP " , 0 ) , /* AMD Hudson */
SND_PCI_QUIRK ( 0x103c , 0x2192 , " HP " , 0 ) , /* AMD Hudson */
SND_PCI_QUIRK ( 0x103c , 0x21f7 , " HP " , 0 ) , /* AMD Hudson */
SND_PCI_QUIRK ( 0x103c , 0x21fa , " HP " , 0 ) , /* AMD Hudson */
2009-12-22 08:15:01 +01:00
SND_PCI_QUIRK ( 0x1043 , 0x81f2 , " ASUS " , 0 ) , /* Athlon64 X2 + nvidia */
2010-02-15 17:05:28 +01:00
SND_PCI_QUIRK ( 0x1043 , 0x81f6 , " ASUS " , 0 ) , /* nvidia */
2010-03-09 18:25:47 +01:00
SND_PCI_QUIRK ( 0x1043 , 0x822d , " ASUS " , 0 ) , /* Athlon64 X2 + nvidia MCP55 */
2013-09-09 10:20:48 +02:00
SND_PCI_QUIRK ( 0x1179 , 0xfb44 , " Toshiba Satellite C870 " , 0 ) , /* AMD Hudson */
2010-03-06 21:06:46 +01:00
SND_PCI_QUIRK ( 0x1849 , 0x0888 , " ASRock " , 0 ) , /* Athlon64 X2 + nvidia */
2010-04-04 12:14:03 +02:00
SND_PCI_QUIRK ( 0xa0a0 , 0x0575 , " Aopen MZ915-M " , 0 ) , /* ICH6 */
2009-08-11 14:21:26 +02:00
{ }
} ;
2012-12-06 12:35:10 -05:00
static void check_msi ( struct azx * chip )
2009-08-11 14:21:26 +02:00
{
const struct snd_pci_quirk * q ;
2009-09-28 13:14:04 +02:00
if ( enable_msi > = 0 ) {
chip - > msi = ! ! enable_msi ;
2009-08-11 14:21:26 +02:00
return ;
2009-09-28 13:14:04 +02:00
}
chip - > msi = 1 ; /* enable MSI as default */
q = snd_pci_quirk_lookup ( chip - > pci , msi_black_list ) ;
2009-08-11 14:21:26 +02:00
if ( q ) {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev ,
" msi for device %04x:%04x set to %d \n " ,
q - > subvendor , q - > subdevice , q - > value ) ;
2009-08-11 14:21:26 +02:00
chip - > msi = q - > value ;
2010-03-15 15:51:53 +01:00
return ;
}
/* NVidia chipsets seem to cause troubles with MSI */
2011-05-25 09:11:37 +02:00
if ( chip - > driver_caps & AZX_DCAPS_NO_MSI ) {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev , " Disabling MSI \n " ) ;
2010-03-15 15:51:53 +01:00
chip - > msi = 0 ;
2009-08-11 14:21:26 +02:00
}
}
2011-12-14 09:27:04 +01:00
/* check the snoop mode availability */
2012-12-06 12:35:10 -05:00
static void azx_check_snoop_available ( struct azx * chip )
2011-12-14 09:27:04 +01:00
{
bool snoop = chip - > snoop ;
switch ( chip - > driver_type ) {
case AZX_DRIVER_VIA :
/* force to non-snoop mode for a new VIA controller
* when BIOS is set
*/
if ( snoop ) {
u8 val ;
pci_read_config_byte ( chip - > pci , 0x42 , & val ) ;
if ( ! ( val & 0x80 ) & & chip - > pci - > revision = = 0x30 )
snoop = false ;
}
break ;
case AZX_DRIVER_ATIHDMI_NS :
/* new ATI HDMI requires non-snoop */
snoop = false ;
break ;
2013-02-07 17:36:22 +01:00
case AZX_DRIVER_CTHDA :
2014-08-06 14:27:42 +02:00
case AZX_DRIVER_CMEDIA :
2013-02-07 17:36:22 +01:00
snoop = false ;
break ;
2011-12-14 09:27:04 +01:00
}
if ( snoop ! = chip - > snoop ) {
2014-02-25 12:21:03 +01:00
dev_info ( chip - > card - > dev , " Force to %s mode \n " ,
snoop ? " snoop " : " non-snoop " ) ;
2011-12-14 09:27:04 +01:00
chip - > snoop = snoop ;
}
}
2007-08-17 09:17:36 +02:00
2013-05-30 22:07:10 +08:00
static void azx_probe_work ( struct work_struct * work )
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( work , struct hda_intel , probe_work ) ;
azx_probe_continue ( & hda - > chip ) ;
2013-05-30 22:07:10 +08:00
}
2005-04-16 15:20:36 -07:00
/*
* constructor
*/
2012-12-06 12:35:10 -05:00
static int azx_create ( struct snd_card * card , struct pci_dev * pci ,
int dev , unsigned int driver_caps ,
2014-02-28 15:41:13 -08:00
const struct hda_controller_ops * hda_ops ,
2012-12-06 12:35:10 -05:00
struct azx * * rchip )
2005-04-16 15:20:36 -07:00
{
2005-11-17 14:59:02 +01:00
static struct snd_device_ops ops = {
2005-04-16 15:20:36 -07:00
. dev_free = azx_dev_free ,
} ;
2014-06-26 18:45:16 +08:00
struct hda_intel * hda ;
2012-04-26 12:23:42 +02:00
struct azx * chip ;
int err ;
2005-04-16 15:20:36 -07:00
* rchip = NULL ;
2008-01-15 11:23:55 +01:00
2006-08-31 17:03:43 +02:00
err = pci_enable_device ( pci ) ;
if ( err < 0 )
2005-04-16 15:20:36 -07:00
return err ;
2014-06-26 18:45:16 +08:00
hda = kzalloc ( sizeof ( * hda ) , GFP_KERNEL ) ;
if ( ! hda ) {
dev_err ( card - > dev , " Cannot allocate hda \n " ) ;
2005-04-16 15:20:36 -07:00
pci_disable_device ( pci ) ;
return - ENOMEM ;
}
2014-06-26 18:45:16 +08:00
chip = & hda - > chip ;
2005-04-16 15:20:36 -07:00
spin_lock_init ( & chip - > reg_lock ) ;
2006-01-16 16:34:20 +01:00
mutex_init ( & chip - > open_mutex ) ;
2005-04-16 15:20:36 -07:00
chip - > card = card ;
chip - > pci = pci ;
2014-02-28 15:41:13 -08:00
chip - > ops = hda_ops ;
2005-04-16 15:20:36 -07:00
chip - > irq = - 1 ;
2011-05-25 09:11:37 +02:00
chip - > driver_caps = driver_caps ;
chip - > driver_type = driver_caps & 0xff ;
2009-08-11 14:21:26 +02:00
check_msi ( chip ) ;
2008-06-10 17:53:34 +02:00
chip - > dev_index = dev ;
2014-02-28 15:41:18 -08:00
chip - > jackpoll_ms = jackpoll_ms ;
2011-11-24 14:31:46 +01:00
INIT_LIST_HEAD ( & chip - > pcm_list ) ;
2014-06-26 17:19:20 +02:00
INIT_WORK ( & hda - > irq_pending_work , azx_irq_pending_work ) ;
INIT_LIST_HEAD ( & hda - > list ) ;
2012-04-26 12:23:42 +02:00
init_vga_switcheroo ( chip ) ;
2014-06-26 17:19:20 +02:00
init_completion ( & hda - > probe_wait ) ;
2005-04-16 15:20:36 -07:00
2014-06-26 16:50:16 +02:00
assign_position_fix ( chip , check_position_fix ( chip , position_fix [ dev ] ) ) ;
2012-02-28 11:58:40 +01:00
2008-01-07 15:16:37 +01:00
check_probe_mask ( chip , dev ) ;
2007-02-01 15:46:50 +01:00
2006-01-12 18:28:44 +01:00
chip - > single_cmd = single_cmd ;
2011-09-28 17:16:09 +02:00
chip - > snoop = hda_snoop ;
2011-12-14 09:27:04 +01:00
azx_check_snoop_available ( chip ) ;
2005-05-12 14:26:27 +02:00
2008-06-10 17:53:35 +02:00
if ( bdl_pos_adj [ dev ] < 0 ) {
switch ( chip - > driver_type ) {
2008-06-13 20:50:27 +02:00
case AZX_DRIVER_ICH :
2010-02-22 17:31:09 -08:00
case AZX_DRIVER_PCH :
2008-06-13 20:50:27 +02:00
bdl_pos_adj [ dev ] = 1 ;
2008-06-10 17:53:35 +02:00
break ;
default :
2008-06-13 20:50:27 +02:00
bdl_pos_adj [ dev ] = 32 ;
2008-06-10 17:53:35 +02:00
break ;
}
}
2014-02-28 15:41:14 -08:00
chip - > bdl_pos_adj = bdl_pos_adj ;
2008-06-10 17:53:35 +02:00
2012-04-26 12:23:42 +02:00
err = snd_device_new ( card , SNDRV_DEV_LOWLEVEL , chip , & ops ) ;
if ( err < 0 ) {
2014-02-25 12:21:03 +01:00
dev_err ( card - > dev , " Error creating device [card]! \n " ) ;
2012-04-26 12:23:42 +02:00
azx_free ( chip ) ;
return err ;
}
2013-05-30 22:07:10 +08:00
/* continue probing in work context as may trigger request module */
2014-06-26 17:19:20 +02:00
INIT_WORK ( & hda - > probe_work , azx_probe_work ) ;
2013-05-30 22:07:10 +08:00
2012-04-26 12:23:42 +02:00
* rchip = chip ;
2013-05-30 22:07:10 +08:00
2012-04-26 12:23:42 +02:00
return 0 ;
}
2012-12-07 07:40:35 +01:00
static int azx_first_init ( struct azx * chip )
2012-04-26 12:23:42 +02:00
{
int dev = chip - > dev_index ;
struct pci_dev * pci = chip - > pci ;
struct snd_card * card = chip - > card ;
2014-02-28 15:41:23 -08:00
int err ;
2012-04-26 12:23:42 +02:00
unsigned short gcap ;
2005-08-24 14:14:57 +02:00
# if BITS_PER_LONG != 64
/* Fix up base address on ULI M5461 */
if ( chip - > driver_type = = AZX_DRIVER_ULI ) {
u16 tmp3 ;
pci_read_config_word ( pci , 0x40 , & tmp3 ) ;
pci_write_config_word ( pci , 0x40 , tmp3 | 0x10 ) ;
pci_write_config_dword ( pci , PCI_BASE_ADDRESS_1 , 0 ) ;
}
# endif
2006-08-31 17:03:43 +02:00
err = pci_request_regions ( pci , " ICH HD audio " ) ;
2012-04-26 12:23:42 +02:00
if ( err < 0 )
2005-04-16 15:20:36 -07:00
return err ;
2012-04-26 12:23:42 +02:00
chip - > region_requested = 1 ;
2005-04-16 15:20:36 -07:00
2006-08-31 17:03:43 +02:00
chip - > addr = pci_resource_start ( pci , 0 ) ;
2008-09-28 16:20:09 -07:00
chip - > remap_addr = pci_ioremap_bar ( pci , 0 ) ;
2005-04-16 15:20:36 -07:00
if ( chip - > remap_addr = = NULL ) {
2014-02-25 12:21:03 +01:00
dev_err ( card - > dev , " ioremap error \n " ) ;
2012-04-26 12:23:42 +02:00
return - ENXIO ;
2005-04-16 15:20:36 -07:00
}
2006-10-23 13:40:59 +02:00
if ( chip - > msi )
if ( pci_enable_msi ( pci ) < 0 )
chip - > msi = 0 ;
2006-08-21 19:17:46 +02:00
2012-04-26 12:23:42 +02:00
if ( azx_acquire_irq ( chip , 0 ) < 0 )
return - EBUSY ;
2005-04-16 15:20:36 -07:00
pci_set_master ( pci ) ;
synchronize_irq ( chip - > irq ) ;
2008-01-15 11:23:55 +01:00
gcap = azx_readw ( chip , GCAP ) ;
2014-02-25 12:21:03 +01:00
dev_dbg ( card - > dev , " chipset global capabilities = 0x%x \n " , gcap ) ;
2008-01-15 11:23:55 +01:00
2009-07-08 13:55:31 +08:00
/* disable SB600 64bit support for safety */
2011-05-25 09:11:37 +02:00
if ( chip - > pci - > vendor = = PCI_VENDOR_ID_ATI ) {
2009-07-08 13:55:31 +08:00
struct pci_dev * p_smbus ;
p_smbus = pci_get_device ( PCI_VENDOR_ID_ATI ,
PCI_DEVICE_ID_ATI_SBX00_SMBUS ,
NULL ) ;
if ( p_smbus ) {
if ( p_smbus - > revision < 0x30 )
2014-06-26 17:54:37 +02:00
gcap & = ~ AZX_GCAP_64OK ;
2009-07-08 13:55:31 +08:00
pci_dev_put ( p_smbus ) ;
}
}
2009-03-17 07:47:18 +01:00
2011-05-25 09:11:37 +02:00
/* disable 64bit DMA address on some devices */
if ( chip - > driver_caps & AZX_DCAPS_NO_64BIT ) {
2014-02-25 12:21:03 +01:00
dev_dbg ( card - > dev , " Disabling 64bit DMA \n " ) ;
2014-06-26 17:54:37 +02:00
gcap & = ~ AZX_GCAP_64OK ;
2011-05-25 09:11:37 +02:00
}
2009-12-09 10:44:47 +01:00
2011-08-04 10:12:56 -05:00
/* disable buffer size rounding to 128-byte multiples if supported */
2012-01-23 17:53:39 +01:00
if ( align_buffer_size > = 0 )
chip - > align_buffer_size = ! ! align_buffer_size ;
else {
if ( chip - > driver_caps & AZX_DCAPS_BUFSIZE )
chip - > align_buffer_size = 0 ;
else if ( chip - > driver_caps & AZX_DCAPS_ALIGN_BUFSIZE )
chip - > align_buffer_size = 1 ;
else
chip - > align_buffer_size = 1 ;
}
2011-08-04 10:12:56 -05:00
2008-02-06 15:05:57 +01:00
/* allow 64bit DMA address if supported by H/W */
2014-06-26 17:54:37 +02:00
if ( ( gcap & AZX_GCAP_64OK ) & & ! pci_set_dma_mask ( pci , DMA_BIT_MASK ( 64 ) ) )
2009-04-13 14:40:14 -07:00
pci_set_consistent_dma_mask ( pci , DMA_BIT_MASK ( 64 ) ) ;
2009-03-17 07:47:18 +01:00
else {
2009-04-13 14:40:14 -07:00
pci_set_dma_mask ( pci , DMA_BIT_MASK ( 32 ) ) ;
pci_set_consistent_dma_mask ( pci , DMA_BIT_MASK ( 32 ) ) ;
2009-03-17 07:47:18 +01:00
}
2008-02-06 15:05:57 +01:00
2008-02-19 11:36:35 +01:00
/* read number of streams from GCAP register instead of using
* hardcoded value
*/
chip - > capture_streams = ( gcap > > 8 ) & 0x0f ;
chip - > playback_streams = ( gcap > > 12 ) & 0x0f ;
if ( ! chip - > playback_streams & & ! chip - > capture_streams ) {
2008-01-15 11:23:55 +01:00
/* gcap didn't give any info, switching to old method */
switch ( chip - > driver_type ) {
case AZX_DRIVER_ULI :
chip - > playback_streams = ULI_NUM_PLAYBACK ;
chip - > capture_streams = ULI_NUM_CAPTURE ;
break ;
case AZX_DRIVER_ATIHDMI :
2011-12-14 16:10:27 +08:00
case AZX_DRIVER_ATIHDMI_NS :
2008-01-15 11:23:55 +01:00
chip - > playback_streams = ATIHDMI_NUM_PLAYBACK ;
chip - > capture_streams = ATIHDMI_NUM_CAPTURE ;
break ;
2008-11-13 11:07:07 +01:00
case AZX_DRIVER_GENERIC :
2008-01-15 11:23:55 +01:00
default :
chip - > playback_streams = ICH6_NUM_PLAYBACK ;
chip - > capture_streams = ICH6_NUM_CAPTURE ;
break ;
}
2005-08-24 14:14:57 +02:00
}
2008-02-19 11:36:35 +01:00
chip - > capture_index_offset = 0 ;
chip - > playback_index_offset = chip - > capture_streams ;
2005-08-24 14:14:57 +02:00
chip - > num_streams = chip - > playback_streams + chip - > capture_streams ;
2007-07-27 16:52:19 +02:00
chip - > azx_dev = kcalloc ( chip - > num_streams , sizeof ( * chip - > azx_dev ) ,
GFP_KERNEL ) ;
2006-08-31 17:03:43 +02:00
if ( ! chip - > azx_dev ) {
2014-02-25 12:21:03 +01:00
dev_err ( card - > dev , " cannot malloc azx_dev \n " ) ;
2012-04-26 12:23:42 +02:00
return - ENOMEM ;
2005-08-24 14:14:57 +02:00
}
2014-02-28 15:41:23 -08:00
err = azx_alloc_stream_pages ( chip ) ;
2009-05-26 15:22:00 +02:00
if ( err < 0 )
2012-04-26 12:23:42 +02:00
return err ;
2005-04-16 15:20:36 -07:00
/* initialize streams */
azx_init_stream ( chip ) ;
/* initialize chip */
2007-08-10 17:21:45 +02:00
azx_init_pci ( chip ) ;
2014-07-03 17:02:23 +08:00
if ( chip - > driver_caps & AZX_DCAPS_I915_POWERWELL )
haswell_set_bclk ( chip ) ;
2010-03-26 11:04:38 +01:00
azx_init_chip ( chip , ( probe_only [ dev ] & 2 ) = = 0 ) ;
2005-04-16 15:20:36 -07:00
/* codec detection */
2006-08-31 17:03:43 +02:00
if ( ! chip - > codec_mask ) {
2014-02-25 12:21:03 +01:00
dev_err ( card - > dev , " no codecs found! \n " ) ;
2012-04-26 12:23:42 +02:00
return - ENODEV ;
2005-04-16 15:20:36 -07:00
}
2005-08-24 14:14:57 +02:00
strcpy ( card - > driver , " HDA-Intel " ) ;
2009-04-16 10:22:24 +02:00
strlcpy ( card - > shortname , driver_short_names [ chip - > driver_type ] ,
sizeof ( card - > shortname ) ) ;
snprintf ( card - > longname , sizeof ( card - > longname ) ,
" %s at 0x%lx irq %i " ,
card - > shortname , chip - > addr , chip - > irq ) ;
2005-08-24 14:14:57 +02:00
2005-04-16 15:20:36 -07:00
return 0 ;
}
2007-08-10 17:21:45 +02:00
static void power_down_all_codecs ( struct azx * chip )
{
2012-08-24 18:38:08 +02:00
# ifdef CONFIG_PM
2007-08-10 17:21:45 +02:00
/* The codecs were powered up in snd_hda_codec_new().
* Now all initialization done , so turn them down if possible
*/
struct hda_codec * codec ;
list_for_each_entry ( codec , & chip - > bus - > codec_list , list ) {
snd_hda_power_down ( codec ) ;
}
# endif
}
2012-08-09 17:40:46 +02:00
# ifdef CONFIG_SND_HDA_PATCH_LOADER
2012-08-09 13:49:23 +02:00
/* callback from request_firmware_nowait() */
static void azx_firmware_cb ( const struct firmware * fw , void * context )
{
struct snd_card * card = context ;
struct azx * chip = card - > private_data ;
struct pci_dev * pci = chip - > pci ;
if ( ! fw ) {
2014-02-25 12:21:03 +01:00
dev_err ( card - > dev , " Cannot load firmware, aborting \n " ) ;
2012-08-09 13:49:23 +02:00
goto error ;
}
chip - > fw = fw ;
if ( ! chip - > disabled ) {
/* continue probing */
if ( azx_probe_continue ( chip ) )
goto error ;
}
return ; /* OK */
error :
snd_card_free ( card ) ;
pci_set_drvdata ( pci , NULL ) ;
}
2012-08-09 17:40:46 +02:00
# endif
2012-08-09 13:49:23 +02:00
2014-02-28 15:41:13 -08:00
/*
* HDA controller ops .
*/
/* PCI register access. */
2014-03-02 20:44:01 -08:00
static void pci_azx_writel ( u32 value , u32 __iomem * addr )
2014-02-28 15:41:13 -08:00
{
writel ( value , addr ) ;
}
2014-03-02 20:44:01 -08:00
static u32 pci_azx_readl ( u32 __iomem * addr )
2014-02-28 15:41:13 -08:00
{
return readl ( addr ) ;
}
2014-03-02 20:44:01 -08:00
static void pci_azx_writew ( u16 value , u16 __iomem * addr )
2014-02-28 15:41:13 -08:00
{
writew ( value , addr ) ;
}
2014-03-02 20:44:01 -08:00
static u16 pci_azx_readw ( u16 __iomem * addr )
2014-02-28 15:41:13 -08:00
{
return readw ( addr ) ;
}
2014-03-02 20:44:01 -08:00
static void pci_azx_writeb ( u8 value , u8 __iomem * addr )
2014-02-28 15:41:13 -08:00
{
writeb ( value , addr ) ;
}
2014-03-02 20:44:01 -08:00
static u8 pci_azx_readb ( u8 __iomem * addr )
2014-02-28 15:41:13 -08:00
{
return readb ( addr ) ;
}
2014-02-28 15:41:16 -08:00
static int disable_msi_reset_irq ( struct azx * chip )
{
int err ;
free_irq ( chip - > irq , chip ) ;
chip - > irq = - 1 ;
pci_disable_msi ( chip - > pci ) ;
chip - > msi = 0 ;
err = azx_acquire_irq ( chip , 1 ) ;
if ( err < 0 )
return err ;
return 0 ;
}
2014-02-28 15:41:20 -08:00
/* DMA page allocation helpers. */
static int dma_alloc_pages ( struct azx * chip ,
int type ,
size_t size ,
struct snd_dma_buffer * buf )
{
int err ;
err = snd_dma_alloc_pages ( type ,
chip - > card - > dev ,
size , buf ) ;
if ( err < 0 )
return err ;
mark_pages_wc ( chip , buf , true ) ;
return 0 ;
}
static void dma_free_pages ( struct azx * chip , struct snd_dma_buffer * buf )
{
mark_pages_wc ( chip , buf , false ) ;
snd_dma_free_pages ( buf ) ;
}
static int substream_alloc_pages ( struct azx * chip ,
struct snd_pcm_substream * substream ,
size_t size )
{
struct azx_dev * azx_dev = get_azx_dev ( substream ) ;
int ret ;
mark_runtime_wc ( chip , azx_dev , substream , false ) ;
azx_dev - > bufsize = 0 ;
azx_dev - > period_bytes = 0 ;
azx_dev - > format_val = 0 ;
ret = snd_pcm_lib_malloc_pages ( substream , size ) ;
if ( ret < 0 )
return ret ;
mark_runtime_wc ( chip , azx_dev , substream , true ) ;
return 0 ;
}
static int substream_free_pages ( struct azx * chip ,
struct snd_pcm_substream * substream )
{
struct azx_dev * azx_dev = get_azx_dev ( substream ) ;
mark_runtime_wc ( chip , azx_dev , substream , false ) ;
return snd_pcm_lib_free_pages ( substream ) ;
}
2014-02-28 15:41:21 -08:00
static void pcm_mmap_prepare ( struct snd_pcm_substream * substream ,
struct vm_area_struct * area )
{
# ifdef CONFIG_X86
struct azx_pcm * apcm = snd_pcm_substream_chip ( substream ) ;
struct azx * chip = apcm - > chip ;
if ( ! azx_snoop ( chip ) )
area - > vm_page_prot = pgprot_writecombine ( area - > vm_page_prot ) ;
# endif
}
2014-02-28 15:41:13 -08:00
static const struct hda_controller_ops pci_hda_ops = {
2014-03-02 20:44:00 -08:00
. reg_writel = pci_azx_writel ,
. reg_readl = pci_azx_readl ,
. reg_writew = pci_azx_writew ,
. reg_readw = pci_azx_readw ,
. reg_writeb = pci_azx_writeb ,
. reg_readb = pci_azx_readb ,
2014-02-28 15:41:16 -08:00
. disable_msi_reset_irq = disable_msi_reset_irq ,
2014-02-28 15:41:20 -08:00
. dma_alloc_pages = dma_alloc_pages ,
. dma_free_pages = dma_free_pages ,
. substream_alloc_pages = substream_alloc_pages ,
. substream_free_pages = substream_free_pages ,
2014-02-28 15:41:21 -08:00
. pcm_mmap_prepare = pcm_mmap_prepare ,
2014-02-28 15:41:28 -08:00
. position_check = azx_position_check ,
2014-02-28 15:41:13 -08:00
} ;
2012-12-06 12:35:10 -05:00
static int azx_probe ( struct pci_dev * pci ,
const struct pci_device_id * pci_id )
2005-04-16 15:20:36 -07:00
{
2008-01-07 15:16:37 +01:00
static int dev ;
2005-11-17 14:59:02 +01:00
struct snd_card * card ;
2014-06-26 17:19:20 +02:00
struct hda_intel * hda ;
2005-11-17 14:59:02 +01:00
struct azx * chip ;
2013-12-02 13:33:57 +01:00
bool schedule_probe ;
2006-08-31 17:03:43 +02:00
int err ;
2005-04-16 15:20:36 -07:00
2008-01-07 15:16:37 +01:00
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 ) {
2014-02-25 12:21:03 +01:00
dev_err ( & pci - > dev , " Error creating card! \n " ) ;
2008-12-28 16:44:30 +01:00
return err ;
2005-04-16 15:20:36 -07:00
}
2014-02-28 15:41:13 -08:00
err = azx_create ( card , pci , dev , pci_id - > driver_data ,
& pci_hda_ops , & chip ) ;
2008-11-20 09:24:52 +08:00
if ( err < 0 )
goto out_free ;
2005-11-17 16:11:09 +01:00
card - > private_data = chip ;
2014-06-26 17:19:20 +02:00
hda = container_of ( chip , struct hda_intel , chip ) ;
2012-12-04 15:09:23 +01:00
pci_set_drvdata ( pci , card ) ;
err = register_vga_switcheroo ( chip ) ;
if ( err < 0 ) {
2014-02-25 12:21:03 +01:00
dev_err ( card - > dev , " Error registering VGA-switcheroo client \n " ) ;
2012-12-04 15:09:23 +01:00
goto out_free ;
}
if ( check_hdmi_disabled ( pci ) ) {
2014-02-25 12:21:03 +01:00
dev_info ( card - > dev , " VGA controller is disabled \n " ) ;
dev_info ( card - > dev , " Delaying initialization \n " ) ;
2012-12-04 15:09:23 +01:00
chip - > disabled = true ;
}
2013-12-02 13:33:57 +01:00
schedule_probe = ! chip - > disabled ;
2005-04-16 15:20:36 -07:00
2012-08-09 12:33:28 +02:00
# ifdef CONFIG_SND_HDA_PATCH_LOADER
if ( patch [ dev ] & & * patch [ dev ] ) {
2014-02-25 12:21:03 +01:00
dev_info ( card - > dev , " Applying patch firmware '%s' \n " ,
patch [ dev ] ) ;
2012-08-09 13:49:23 +02:00
err = request_firmware_nowait ( THIS_MODULE , true , patch [ dev ] ,
& pci - > dev , GFP_KERNEL , card ,
azx_firmware_cb ) ;
2012-08-09 12:33:28 +02:00
if ( err < 0 )
goto out_free ;
2013-12-02 13:33:57 +01:00
schedule_probe = false ; /* continued in azx_firmware_cb() */
2012-08-09 12:33:28 +02:00
}
# endif /* CONFIG_SND_HDA_PATCH_LOADER */
2013-12-02 13:33:57 +01:00
# ifndef CONFIG_SND_HDA_I915
if ( chip - > driver_caps & AZX_DCAPS_I915_POWERWELL )
2014-02-25 12:21:03 +01:00
dev_err ( card - > dev , " Haswell must build in CONFIG_SND_HDA_I915 \n " ) ;
2013-05-30 22:07:10 +08:00
# endif
2013-12-02 13:33:57 +01:00
if ( schedule_probe )
2014-06-26 17:19:20 +02:00
schedule_work ( & hda - > probe_work ) ;
2012-04-26 12:23:42 +02:00
dev + + ;
2013-12-02 11:12:28 +01:00
if ( chip - > disabled )
2014-06-26 17:19:20 +02:00
complete_all ( & hda - > probe_wait ) ;
2012-04-26 12:23:42 +02:00
return 0 ;
out_free :
snd_card_free ( card ) ;
return err ;
}
2014-02-28 15:41:19 -08:00
/* number of codec slots for each chipset: 0 = default slots (i.e. 4) */
static unsigned int azx_max_codecs [ AZX_NUM_DRIVERS ] = {
[ AZX_DRIVER_NVIDIA ] = 8 ,
[ AZX_DRIVER_TERA ] = 1 ,
} ;
2012-12-07 07:40:35 +01:00
static int azx_probe_continue ( struct azx * chip )
2012-04-26 12:23:42 +02:00
{
2014-06-26 17:19:20 +02:00
struct hda_intel * hda = container_of ( chip , struct hda_intel , chip ) ;
2013-05-30 22:07:08 +08:00
struct pci_dev * pci = chip - > pci ;
2012-04-26 12:23:42 +02:00
int dev = chip - > dev_index ;
int err ;
2013-05-30 22:07:10 +08:00
/* Request power well for Haswell HDA controller and codec */
if ( chip - > driver_caps & AZX_DCAPS_I915_POWERWELL ) {
2013-08-19 13:32:30 +02:00
# ifdef CONFIG_SND_HDA_I915
2013-05-30 22:07:10 +08:00
err = hda_i915_init ( ) ;
if ( err < 0 ) {
2014-02-25 12:21:03 +01:00
dev_err ( chip - > card - > dev ,
" Error request power-well from i915 \n " ) ;
2013-05-30 22:07:10 +08:00
goto out_free ;
}
2014-06-13 15:14:34 +02:00
err = hda_display_power ( true ) ;
if ( err < 0 ) {
dev_err ( chip - > card - > dev ,
" Cannot turn on display power on i915 \n " ) ;
goto out_free ;
}
2013-08-19 13:32:30 +02:00
# endif
2013-05-30 22:07:10 +08:00
}
2013-05-30 22:07:09 +08:00
err = azx_first_init ( chip ) ;
if ( err < 0 )
goto out_free ;
2009-11-13 18:41:52 +01:00
# ifdef CONFIG_SND_HDA_INPUT_BEEP
chip - > beep_mode = beep_mode [ dev ] ;
# endif
2005-04-16 15:20:36 -07:00
/* create codec instances */
2014-02-28 15:41:19 -08:00
err = azx_codec_create ( chip , model [ dev ] ,
azx_max_codecs [ chip - > driver_type ] ,
power_save_addr ) ;
2008-11-20 09:24:52 +08:00
if ( err < 0 )
goto out_free ;
2009-06-17 09:52:54 +02:00
# ifdef CONFIG_SND_HDA_PATCH_LOADER
2012-08-09 12:33:28 +02:00
if ( chip - > fw ) {
err = snd_hda_load_patch ( chip - > bus , chip - > fw - > size ,
chip - > fw - > data ) ;
2009-06-17 09:52:54 +02:00
if ( err < 0 )
goto out_free ;
2012-11-22 16:18:13 +01:00
# ifndef CONFIG_PM
2012-08-09 12:33:28 +02:00
release_firmware ( chip - > fw ) ; /* no longer needed */
chip - > fw = NULL ;
2012-11-22 16:18:13 +01:00
# endif
2009-06-17 09:52:54 +02:00
}
# endif
2010-03-26 11:04:38 +01:00
if ( ( probe_only [ dev ] & 1 ) = = 0 ) {
2009-06-17 09:33:52 +02:00
err = azx_codec_configure ( chip ) ;
if ( err < 0 )
goto out_free ;
}
2005-04-16 15:20:36 -07:00
/* create PCM streams */
2008-07-30 15:01:44 +02:00
err = snd_hda_build_pcms ( chip - > bus ) ;
2008-11-20 09:24:52 +08:00
if ( err < 0 )
goto out_free ;
2005-04-16 15:20:36 -07:00
/* create mixer controls */
2007-07-27 16:52:19 +02:00
err = azx_mixer_create ( chip ) ;
2008-11-20 09:24:52 +08:00
if ( err < 0 )
goto out_free ;
2005-04-16 15:20:36 -07:00
2012-04-26 12:23:42 +02:00
err = snd_card_register ( chip - > card ) ;
2008-11-20 09:24:52 +08:00
if ( err < 0 )
goto out_free ;
2005-04-16 15:20:36 -07:00
2007-08-10 17:21:45 +02:00
chip - > running = 1 ;
power_down_all_codecs ( chip ) ;
2008-10-29 16:18:25 +01:00
azx_notifier_register ( chip ) ;
2012-08-14 17:13:32 +02:00
azx_add_card_list ( chip ) ;
2014-06-26 17:19:20 +02:00
if ( ( chip - > driver_caps & AZX_DCAPS_PM_RUNTIME ) | | hda - > use_vga_switcheroo )
2013-05-30 22:07:08 +08:00
pm_runtime_put_noidle ( & pci - > dev ) ;
2005-04-16 15:20:36 -07:00
2008-11-20 09:24:52 +08:00
out_free :
2013-12-02 11:12:28 +01:00
if ( err < 0 )
2014-06-26 17:19:20 +02:00
hda - > init_failed = 1 ;
complete_all ( & hda - > probe_wait ) ;
2008-11-20 09:24:52 +08:00
return err ;
2005-04-16 15:20:36 -07:00
}
2012-12-06 12:35:10 -05:00
static void azx_remove ( struct pci_dev * pci )
2005-04-16 15:20:36 -07:00
{
2012-04-26 12:13:25 +02:00
struct snd_card * card = pci_get_drvdata ( pci ) ;
2012-08-23 17:32:30 +08:00
2012-04-26 12:13:25 +02:00
if ( card )
snd_card_free ( card ) ;
2005-04-16 15:20:36 -07:00
}
/* PCI IDs */
2014-05-22 17:08:54 +02:00
static const struct pci_device_id azx_ids [ ] = {
2010-01-12 17:03:35 -08:00
/* CPT */
2011-05-25 09:11:37 +02:00
{ PCI_DEVICE ( 0x8086 , 0x1c20 ) ,
2013-01-08 13:51:30 +01:00
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH_NOPM } ,
2010-09-10 16:29:56 -07:00
/* PBG */
2011-05-25 09:11:37 +02:00
{ PCI_DEVICE ( 0x8086 , 0x1d20 ) ,
2013-01-08 13:51:30 +01:00
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH_NOPM } ,
2011-04-20 10:59:57 -07:00
/* Panther Point */
2011-05-25 09:11:37 +02:00
{ PCI_DEVICE ( 0x8086 , 0x1e20 ) ,
2013-11-22 12:43:25 +01:00
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
2012-01-23 16:24:31 -08:00
/* Lynx Point */
{ PCI_DEVICE ( 0x8086 , 0x8c20 ) ,
2012-11-19 20:03:37 +01:00
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
2014-05-23 09:02:44 +02:00
/* 9 Series */
{ PCI_DEVICE ( 0x8086 , 0x8ca0 ) ,
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
2013-02-08 17:29:40 -08:00
/* Wellsburg */
{ PCI_DEVICE ( 0x8086 , 0x8d20 ) ,
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
{ PCI_DEVICE ( 0x8086 , 0x8d21 ) ,
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
2012-08-09 09:38:59 -07:00
/* Lynx Point-LP */
{ PCI_DEVICE ( 0x8086 , 0x9c20 ) ,
2012-11-19 20:03:37 +01:00
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
2012-08-09 09:38:59 -07:00
/* Lynx Point-LP */
{ PCI_DEVICE ( 0x8086 , 0x9c21 ) ,
2012-11-19 20:03:37 +01:00
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
2013-11-04 09:27:45 -08:00
/* Wildcat Point-LP */
{ PCI_DEVICE ( 0x8086 , 0x9ca0 ) ,
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
2012-06-13 10:23:51 +08:00
/* Haswell */
2013-02-01 22:42:19 +08:00
{ PCI_DEVICE ( 0x8086 , 0x0a0c ) ,
2013-11-05 17:54:05 +01:00
. driver_data = AZX_DRIVER_HDMI | AZX_DCAPS_INTEL_HASWELL } ,
2012-06-13 10:23:51 +08:00
{ PCI_DEVICE ( 0x8086 , 0x0c0c ) ,
2013-11-05 17:54:05 +01:00
. driver_data = AZX_DRIVER_HDMI | AZX_DCAPS_INTEL_HASWELL } ,
2012-09-17 13:10:23 +08:00
{ PCI_DEVICE ( 0x8086 , 0x0d0c ) ,
2013-11-05 17:54:05 +01:00
. driver_data = AZX_DRIVER_HDMI | AZX_DCAPS_INTEL_HASWELL } ,
2014-01-08 15:55:14 -05:00
/* Broadwell */
{ PCI_DEVICE ( 0x8086 , 0x160c ) ,
2014-06-09 15:28:59 +08:00
. driver_data = AZX_DRIVER_HDMI | AZX_DCAPS_INTEL_BROADWELL } ,
2012-09-21 18:39:07 -05:00
/* 5 Series/3400 */
{ PCI_DEVICE ( 0x8086 , 0x3b56 ) ,
2013-02-14 09:44:55 +01:00
. driver_data = AZX_DRIVER_SCH | AZX_DCAPS_INTEL_PCH_NOPM } ,
2013-01-29 10:12:23 +01:00
/* Poulsbo */
2011-05-25 09:11:37 +02:00
{ PCI_DEVICE ( 0x8086 , 0x811b ) ,
2013-01-29 10:12:23 +01:00
. driver_data = AZX_DRIVER_SCH | AZX_DCAPS_INTEL_PCH_NOPM } ,
/* Oaktrail */
2011-12-28 15:17:26 +00:00
{ PCI_DEVICE ( 0x8086 , 0x080a ) ,
2013-01-29 10:12:23 +01:00
. driver_data = AZX_DRIVER_SCH | AZX_DCAPS_INTEL_PCH_NOPM } ,
2013-05-16 15:36:12 +08:00
/* BayTrail */
{ PCI_DEVICE ( 0x8086 , 0x0f04 ) ,
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH_NOPM } ,
2014-08-04 09:22:44 +08:00
/* Braswell */
{ PCI_DEVICE ( 0x8086 , 0x2284 ) ,
. driver_data = AZX_DRIVER_PCH | AZX_DCAPS_INTEL_PCH } ,
2011-12-14 15:52:30 +08:00
/* ICH */
2011-06-10 14:56:26 +02:00
{ PCI_DEVICE ( 0x8086 , 0x2668 ) ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_OLD_SSYNC |
AZX_DCAPS_BUFSIZE } , /* ICH6 */
2011-06-10 14:56:26 +02:00
{ PCI_DEVICE ( 0x8086 , 0x27d8 ) ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_OLD_SSYNC |
AZX_DCAPS_BUFSIZE } , /* ICH7 */
2011-06-10 14:56:26 +02:00
{ PCI_DEVICE ( 0x8086 , 0x269a ) ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_OLD_SSYNC |
AZX_DCAPS_BUFSIZE } , /* ESB2 */
2011-06-10 14:56:26 +02:00
{ PCI_DEVICE ( 0x8086 , 0x284b ) ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_OLD_SSYNC |
AZX_DCAPS_BUFSIZE } , /* ICH8 */
2011-06-10 14:56:26 +02:00
{ PCI_DEVICE ( 0x8086 , 0x293e ) ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_OLD_SSYNC |
AZX_DCAPS_BUFSIZE } , /* ICH9 */
2011-06-10 14:56:26 +02:00
{ PCI_DEVICE ( 0x8086 , 0x293f ) ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_OLD_SSYNC |
AZX_DCAPS_BUFSIZE } , /* ICH9 */
2011-06-10 14:56:26 +02:00
{ PCI_DEVICE ( 0x8086 , 0x3a3e ) ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_OLD_SSYNC |
AZX_DCAPS_BUFSIZE } , /* ICH10 */
2011-06-10 14:56:26 +02:00
{ PCI_DEVICE ( 0x8086 , 0x3a6e ) ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_OLD_SSYNC |
AZX_DCAPS_BUFSIZE } , /* ICH10 */
2010-09-15 10:17:26 +02:00
/* Generic Intel */
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_ANY_ID ) ,
. class = PCI_CLASS_MULTIMEDIA_HD_AUDIO < < 8 ,
. class_mask = 0xffffff ,
2011-08-04 10:12:56 -05:00
. driver_data = AZX_DRIVER_ICH | AZX_DCAPS_BUFSIZE } ,
2011-05-25 09:11:37 +02:00
/* ATI SB 450/600/700/800/900 */
{ PCI_DEVICE ( 0x1002 , 0x437b ) ,
. driver_data = AZX_DRIVER_ATI | AZX_DCAPS_PRESET_ATI_SB } ,
{ PCI_DEVICE ( 0x1002 , 0x4383 ) ,
. driver_data = AZX_DRIVER_ATI | AZX_DCAPS_PRESET_ATI_SB } ,
/* AMD Hudson */
{ PCI_DEVICE ( 0x1022 , 0x780d ) ,
. driver_data = AZX_DRIVER_GENERIC | AZX_DCAPS_PRESET_ATI_SB } ,
2008-02-21 08:13:11 +01:00
/* ATI HDMI */
2011-05-25 09:11:37 +02:00
{ PCI_DEVICE ( 0x1002 , 0x793b ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0x7919 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0x960f ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0x970f ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa00 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa08 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa10 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa18 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa20 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa28 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa30 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa38 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa40 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa48 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
2013-11-05 09:27:10 +01:00
{ PCI_DEVICE ( 0x1002 , 0xaa50 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa58 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa60 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa68 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa80 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa88 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa90 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaa98 ) ,
. driver_data = AZX_DRIVER_ATIHDMI | AZX_DCAPS_PRESET_ATI_HDMI } ,
2011-12-14 16:10:27 +08:00
{ PCI_DEVICE ( 0x1002 , 0x9902 ) ,
. driver_data = AZX_DRIVER_ATIHDMI_NS | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaaa0 ) ,
. driver_data = AZX_DRIVER_ATIHDMI_NS | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaaa8 ) ,
. driver_data = AZX_DRIVER_ATIHDMI_NS | AZX_DCAPS_PRESET_ATI_HDMI } ,
{ PCI_DEVICE ( 0x1002 , 0xaab0 ) ,
. driver_data = AZX_DRIVER_ATIHDMI_NS | AZX_DCAPS_PRESET_ATI_HDMI } ,
2008-02-21 08:13:11 +01:00
/* VIA VT8251/VT8237A */
2011-05-25 09:11:37 +02:00
{ PCI_DEVICE ( 0x1106 , 0x3288 ) ,
. driver_data = AZX_DRIVER_VIA | AZX_DCAPS_POSFIX_VIA } ,
2012-06-08 19:18:39 +08:00
/* VIA GFX VT7122/VX900 */
{ PCI_DEVICE ( 0x1106 , 0x9170 ) , . driver_data = AZX_DRIVER_GENERIC } ,
/* VIA GFX VT6122/VX11 */
{ PCI_DEVICE ( 0x1106 , 0x9140 ) , . driver_data = AZX_DRIVER_GENERIC } ,
2008-02-21 08:13:11 +01:00
/* SIS966 */
{ PCI_DEVICE ( 0x1039 , 0x7502 ) , . driver_data = AZX_DRIVER_SIS } ,
/* ULI M5461 */
{ PCI_DEVICE ( 0x10b9 , 0x5461 ) , . driver_data = AZX_DRIVER_ULI } ,
/* NVIDIA MCP */
2009-12-18 16:41:39 +01:00
{ PCI_DEVICE ( PCI_VENDOR_ID_NVIDIA , PCI_ANY_ID ) ,
. class = PCI_CLASS_MULTIMEDIA_HD_AUDIO < < 8 ,
. class_mask = 0xffffff ,
2011-05-25 09:11:37 +02:00
. driver_data = AZX_DRIVER_NVIDIA | AZX_DCAPS_PRESET_NVIDIA } ,
2008-05-27 11:44:55 +02:00
/* Teradici */
2011-05-25 09:11:37 +02:00
{ PCI_DEVICE ( 0x6549 , 0x1200 ) ,
. driver_data = AZX_DRIVER_TERA | AZX_DCAPS_NO_64BIT } ,
2012-11-02 13:10:39 -07:00
{ PCI_DEVICE ( 0x6549 , 0x2200 ) ,
. driver_data = AZX_DRIVER_TERA | AZX_DCAPS_NO_64BIT } ,
2009-04-16 08:53:34 +02:00
/* Creative X-Fi (CA0110-IBG) */
2012-06-11 15:51:54 +02:00
/* CTHDA chips */
{ PCI_DEVICE ( 0x1102 , 0x0010 ) ,
. driver_data = AZX_DRIVER_CTHDA | AZX_DCAPS_PRESET_CTHDA } ,
{ PCI_DEVICE ( 0x1102 , 0x0012 ) ,
. driver_data = AZX_DRIVER_CTHDA | AZX_DCAPS_PRESET_CTHDA } ,
2014-02-10 09:48:47 +01:00
# if !IS_ENABLED(CONFIG_SND_CTXFI)
2009-05-18 12:40:52 +02:00
/* the following entry conflicts with snd-ctxfi driver,
* as ctxfi driver mutates from HD - audio to native mode with
* a special command sequence .
*/
2009-04-16 08:53:34 +02:00
{ PCI_DEVICE ( PCI_VENDOR_ID_CREATIVE , PCI_ANY_ID ) ,
. class = PCI_CLASS_MULTIMEDIA_HD_AUDIO < < 8 ,
. class_mask = 0xffffff ,
2011-05-25 09:11:37 +02:00
. driver_data = AZX_DRIVER_CTX | AZX_DCAPS_CTX_WORKAROUND |
2011-11-06 13:49:13 +01:00
AZX_DCAPS_RIRB_PRE_DELAY | AZX_DCAPS_POSFIX_LPIB } ,
2009-05-18 12:40:52 +02:00
# else
/* this entry seems still valid -- i.e. without emu20kx chip */
2011-05-25 09:11:37 +02:00
{ PCI_DEVICE ( 0x1102 , 0x0009 ) ,
. driver_data = AZX_DRIVER_CTX | AZX_DCAPS_CTX_WORKAROUND |
2011-11-06 13:49:13 +01:00
AZX_DCAPS_RIRB_PRE_DELAY | AZX_DCAPS_POSFIX_LPIB } ,
2009-05-18 12:40:52 +02:00
# endif
2014-08-06 14:27:42 +02:00
/* CM8888 */
{ PCI_DEVICE ( 0x13f6 , 0x5011 ) ,
. driver_data = AZX_DRIVER_CMEDIA |
AZX_DCAPS_NO_MSI | AZX_DCAPS_POSFIX_LPIB } ,
2010-09-26 23:35:06 -03:00
/* Vortex86MX */
{ PCI_DEVICE ( 0x17f3 , 0x3010 ) , . driver_data = AZX_DRIVER_GENERIC } ,
2011-01-17 15:23:21 +01:00
/* VMware HDAudio */
{ PCI_DEVICE ( 0x15ad , 0x1977 ) , . driver_data = AZX_DRIVER_GENERIC } ,
2009-07-17 11:32:32 +08:00
/* AMD/ATI Generic, PCI class code and Vendor ID for HD Audio */
2008-11-13 11:07:07 +01:00
{ PCI_DEVICE ( PCI_VENDOR_ID_ATI , PCI_ANY_ID ) ,
. class = PCI_CLASS_MULTIMEDIA_HD_AUDIO < < 8 ,
. class_mask = 0xffffff ,
2011-05-25 09:11:37 +02:00
. driver_data = AZX_DRIVER_GENERIC | AZX_DCAPS_PRESET_ATI_HDMI } ,
2009-07-17 11:32:32 +08:00
{ PCI_DEVICE ( PCI_VENDOR_ID_AMD , PCI_ANY_ID ) ,
. class = PCI_CLASS_MULTIMEDIA_HD_AUDIO < < 8 ,
. class_mask = 0xffffff ,
2011-05-25 09:11:37 +02:00
. driver_data = AZX_DRIVER_GENERIC | AZX_DCAPS_PRESET_ATI_HDMI } ,
2005-04-16 15:20:36 -07:00
{ 0 , }
} ;
MODULE_DEVICE_TABLE ( pci , azx_ids ) ;
/* pci_driver definition */
2012-04-24 12:25:00 +02:00
static struct pci_driver azx_driver = {
2011-06-10 16:20:20 +02:00
. name = KBUILD_MODNAME ,
2005-04-16 15:20:36 -07:00
. id_table = azx_ids ,
. probe = azx_probe ,
2012-12-06 12:35:10 -05:00
. remove = azx_remove ,
2012-07-02 15:20:37 +02:00
. driver = {
. pm = AZX_PM_OPS ,
} ,
2005-04-16 15:20:36 -07:00
} ;
2012-04-24 12:25:00 +02:00
module_pci_driver ( azx_driver ) ;