2005-04-17 02:20:36 +04:00
/*
* PMac AWACS lowlevel functions
*
* Copyright ( c ) by Takashi Iwai < tiwai @ suse . de >
* code based on dmasound . c .
*
* 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 <asm/io.h>
# include <asm/nvram.h>
# include <linux/init.h>
# include <linux/delay.h>
# include <linux/slab.h>
# include <sound/core.h>
# include "pmac.h"
# ifdef CONFIG_ADB_CUDA
# define PMAC_AMP_AVAIL
# endif
# ifdef PMAC_AMP_AVAIL
2005-11-17 17:09:46 +03:00
struct awacs_amp {
2005-04-17 02:20:36 +04:00
unsigned char amp_master ;
unsigned char amp_vol [ 2 ] [ 2 ] ;
unsigned char amp_tone [ 2 ] ;
2005-11-17 17:09:46 +03:00
} ;
2005-04-17 02:20:36 +04:00
# define CHECK_CUDA_AMP() (sys_ctrler == SYS_CTRLER_CUDA)
# endif /* PMAC_AMP_AVAIL */
2005-11-17 17:09:46 +03:00
static void snd_pmac_screamer_wait ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
long timeout = 2000 ;
while ( ! ( in_le32 ( & chip - > awacs - > codec_stat ) & MASK_VALID ) ) {
mdelay ( 1 ) ;
if ( ! - - timeout ) {
snd_printd ( " snd_pmac_screamer_wait timeout \n " ) ;
break ;
}
}
}
/*
* write AWACS register
*/
static void
2005-11-17 17:09:46 +03:00
snd_pmac_awacs_write ( struct snd_pmac * chip , int val )
2005-04-17 02:20:36 +04:00
{
long timeout = 5000000 ;
if ( chip - > model = = PMAC_SCREAMER )
snd_pmac_screamer_wait ( chip ) ;
out_le32 ( & chip - > awacs - > codec_ctrl , val | ( chip - > subframe < < 22 ) ) ;
while ( in_le32 ( & chip - > awacs - > codec_ctrl ) & MASK_NEWECMD ) {
if ( ! - - timeout ) {
snd_printd ( " snd_pmac_awacs_write timeout \n " ) ;
break ;
}
}
}
static void
2005-11-17 17:09:46 +03:00
snd_pmac_awacs_write_reg ( struct snd_pmac * chip , int reg , int val )
2005-04-17 02:20:36 +04:00
{
snd_pmac_awacs_write ( chip , val | ( reg < < 12 ) ) ;
chip - > awacs_reg [ reg ] = val ;
}
static void
2005-11-17 17:09:46 +03:00
snd_pmac_awacs_write_noreg ( struct snd_pmac * chip , int reg , int val )
2005-04-17 02:20:36 +04:00
{
snd_pmac_awacs_write ( chip , val | ( reg < < 12 ) ) ;
}
2005-06-28 01:36:34 +04:00
# ifdef CONFIG_PM
2005-04-17 02:20:36 +04:00
/* Recalibrate chip */
2005-11-17 17:09:46 +03:00
static void screamer_recalibrate ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
if ( chip - > model ! = PMAC_SCREAMER )
return ;
/* Sorry for the horrible delays... I hope to get that improved
* by making the whole PM process asynchronous in a future version
*/
snd_pmac_awacs_write_noreg ( chip , 1 , chip - > awacs_reg [ 1 ] ) ;
if ( chip - > manufacturer = = 0x1 )
/* delay for broken crystal part */
2005-07-09 12:53:24 +04:00
msleep ( 750 ) ;
2005-04-17 02:20:36 +04:00
snd_pmac_awacs_write_noreg ( chip , 1 ,
2005-11-17 17:09:46 +03:00
chip - > awacs_reg [ 1 ] | MASK_RECALIBRATE |
MASK_CMUTE | MASK_AMUTE ) ;
2005-04-17 02:20:36 +04:00
snd_pmac_awacs_write_noreg ( chip , 1 , chip - > awacs_reg [ 1 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 6 , chip - > awacs_reg [ 6 ] ) ;
}
# else
# define screamer_recalibrate(chip) /* NOP */
# endif
/*
* additional callback to set the pcm format
*/
2005-11-17 17:09:46 +03:00
static void snd_pmac_awacs_set_format ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
chip - > awacs_reg [ 1 ] & = ~ MASK_SAMPLERATE ;
chip - > awacs_reg [ 1 ] | = chip - > rate_index < < 3 ;
snd_pmac_awacs_write_reg ( chip , 1 , chip - > awacs_reg [ 1 ] ) ;
}
/*
* AWACS volume callbacks
*/
/*
* volumes : 0 - 15 stereo
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_info_volume ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 2 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 15 ;
return 0 ;
}
2008-04-16 21:39:27 +04:00
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_volume ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int reg = kcontrol - > private_value & 0xff ;
int lshift = ( kcontrol - > private_value > > 8 ) & 0xff ;
int inverted = ( kcontrol - > private_value > > 16 ) & 1 ;
unsigned long flags ;
int vol [ 2 ] ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
vol [ 0 ] = ( chip - > awacs_reg [ reg ] > > lshift ) & 0xf ;
vol [ 1 ] = chip - > awacs_reg [ reg ] & 0xf ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
if ( inverted ) {
vol [ 0 ] = 0x0f - vol [ 0 ] ;
vol [ 1 ] = 0x0f - vol [ 1 ] ;
}
ucontrol - > value . integer . value [ 0 ] = vol [ 0 ] ;
ucontrol - > value . integer . value [ 1 ] = vol [ 1 ] ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_volume ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int reg = kcontrol - > private_value & 0xff ;
int lshift = ( kcontrol - > private_value > > 8 ) & 0xff ;
int inverted = ( kcontrol - > private_value > > 16 ) & 1 ;
int val , oldval ;
unsigned long flags ;
2007-11-15 18:14:12 +03:00
unsigned int vol [ 2 ] ;
2005-04-17 02:20:36 +04:00
vol [ 0 ] = ucontrol - > value . integer . value [ 0 ] ;
vol [ 1 ] = ucontrol - > value . integer . value [ 1 ] ;
2007-11-15 18:14:12 +03:00
if ( vol [ 0 ] > 0x0f | | vol [ 1 ] > 0x0f )
return - EINVAL ;
2005-04-17 02:20:36 +04:00
if ( inverted ) {
vol [ 0 ] = 0x0f - vol [ 0 ] ;
vol [ 1 ] = 0x0f - vol [ 1 ] ;
}
vol [ 0 ] & = 0x0f ;
vol [ 1 ] & = 0x0f ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
oldval = chip - > awacs_reg [ reg ] ;
val = oldval & ~ ( 0xf | ( 0xf < < lshift ) ) ;
val | = vol [ 0 ] < < lshift ;
val | = vol [ 1 ] ;
if ( oldval ! = val )
snd_pmac_awacs_write_reg ( chip , reg , val ) ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
return oldval ! = reg ;
}
# define AWACS_VOLUME(xname, xreg, xshift, xinverted) \
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER , . name = xname , . index = 0 , \
. info = snd_pmac_awacs_info_volume , \
. get = snd_pmac_awacs_get_volume , \
. put = snd_pmac_awacs_put_volume , \
. private_value = ( xreg ) | ( ( xshift ) < < 8 ) | ( ( xinverted ) < < 16 ) }
/*
* mute master / ogain for AWACS : mono
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_switch ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int reg = kcontrol - > private_value & 0xff ;
int shift = ( kcontrol - > private_value > > 8 ) & 0xff ;
int invert = ( kcontrol - > private_value > > 16 ) & 1 ;
int val ;
unsigned long flags ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
val = ( chip - > awacs_reg [ reg ] > > shift ) & 1 ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
if ( invert )
val = 1 - val ;
ucontrol - > value . integer . value [ 0 ] = val ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_switch ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int reg = kcontrol - > private_value & 0xff ;
int shift = ( kcontrol - > private_value > > 8 ) & 0xff ;
int invert = ( kcontrol - > private_value > > 16 ) & 1 ;
int mask = 1 < < shift ;
int val , changed ;
unsigned long flags ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
val = chip - > awacs_reg [ reg ] & ~ mask ;
if ( ucontrol - > value . integer . value [ 0 ] ! = invert )
val | = mask ;
changed = chip - > awacs_reg [ reg ] ! = val ;
if ( changed )
snd_pmac_awacs_write_reg ( chip , reg , val ) ;
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
return changed ;
}
# define AWACS_SWITCH(xname, xreg, xshift, xinvert) \
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER , . name = xname , . index = 0 , \
. info = snd_pmac_boolean_mono_info , \
. get = snd_pmac_awacs_get_switch , \
. put = snd_pmac_awacs_put_switch , \
. private_value = ( xreg ) | ( ( xshift ) < < 8 ) | ( ( xinvert ) < < 16 ) }
# ifdef PMAC_AMP_AVAIL
/*
* controls for perch / whisper extension cards , e . g . G3 desktop
*
* TDA7433 connected via i2c address 0x45 ( = 0x8a ) ,
* accessed through cuda
*/
static void awacs_set_cuda ( int reg , int val )
{
struct adb_request req ;
2008-04-16 21:39:27 +04:00
cuda_request ( & req , NULL , 5 , CUDA_PACKET , CUDA_GET_SET_IIC , 0x8a ,
reg , val ) ;
2005-04-17 02:20:36 +04:00
while ( ! req . complete )
cuda_poll ( ) ;
}
/*
* level = 0 - 14 , 7 = 0 dB
*/
2005-11-17 17:09:46 +03:00
static void awacs_amp_set_tone ( struct awacs_amp * amp , int bass , int treble )
2005-04-17 02:20:36 +04:00
{
amp - > amp_tone [ 0 ] = bass ;
amp - > amp_tone [ 1 ] = treble ;
if ( bass > 7 )
bass = ( 14 - bass ) + 8 ;
if ( treble > 7 )
treble = ( 14 - treble ) + 8 ;
awacs_set_cuda ( 2 , ( bass < < 4 ) | treble ) ;
}
/*
* vol = 0 - 31 ( attenuation ) , 32 = mute bit , stereo
*/
2008-04-16 21:39:27 +04:00
static int awacs_amp_set_vol ( struct awacs_amp * amp , int index ,
int lvol , int rvol , int do_check )
2005-04-17 02:20:36 +04:00
{
if ( do_check & & amp - > amp_vol [ index ] [ 0 ] = = lvol & &
2008-04-16 21:39:27 +04:00
amp - > amp_vol [ index ] [ 1 ] = = rvol )
2005-04-17 02:20:36 +04:00
return 0 ;
awacs_set_cuda ( 3 + index , lvol ) ;
awacs_set_cuda ( 5 + index , rvol ) ;
amp - > amp_vol [ index ] [ 0 ] = lvol ;
amp - > amp_vol [ index ] [ 1 ] = rvol ;
return 1 ;
}
/*
* 0 = - 79 dB , 79 = 0 dB , 99 = + 20 dB
*/
2005-11-17 17:09:46 +03:00
static void awacs_amp_set_master ( struct awacs_amp * amp , int vol )
2005-04-17 02:20:36 +04:00
{
amp - > amp_master = vol ;
if ( vol < = 79 )
vol = 32 + ( 79 - vol ) ;
else
vol = 32 - ( vol - 79 ) ;
awacs_set_cuda ( 1 , vol ) ;
}
2005-11-17 17:09:46 +03:00
static void awacs_amp_free ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2008-08-08 19:12:47 +04:00
if ( ! amp )
return ;
2005-04-17 02:20:36 +04:00
kfree ( amp ) ;
chip - > mixer_data = NULL ;
chip - > mixer_free = NULL ;
}
/*
* mixer controls
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_info_volume_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 2 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 31 ;
return 0 ;
}
2008-04-16 21:39:27 +04:00
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_volume_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2008-08-08 19:12:47 +04:00
2005-04-17 02:20:36 +04:00
ucontrol - > value . integer . value [ 0 ] = 31 - ( amp - > amp_vol [ index ] [ 0 ] & 31 ) ;
ucontrol - > value . integer . value [ 1 ] = 31 - ( amp - > amp_vol [ index ] [ 1 ] & 31 ) ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_volume_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
int vol [ 2 ] ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
2008-04-16 21:39:27 +04:00
vol [ 0 ] = ( 31 - ( ucontrol - > value . integer . value [ 0 ] & 31 ) )
| ( amp - > amp_vol [ index ] [ 0 ] & 32 ) ;
vol [ 1 ] = ( 31 - ( ucontrol - > value . integer . value [ 1 ] & 31 ) )
| ( amp - > amp_vol [ index ] [ 1 ] & 32 ) ;
2005-04-17 02:20:36 +04:00
return awacs_amp_set_vol ( amp , index , vol [ 0 ] , vol [ 1 ] , 1 ) ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_switch_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2008-08-08 19:12:47 +04:00
2008-04-16 21:39:27 +04:00
ucontrol - > value . integer . value [ 0 ] = ( amp - > amp_vol [ index ] [ 0 ] & 32 )
? 0 : 1 ;
ucontrol - > value . integer . value [ 1 ] = ( amp - > amp_vol [ index ] [ 1 ] & 32 )
? 0 : 1 ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_switch_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
int vol [ 2 ] ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
2008-04-16 21:39:27 +04:00
vol [ 0 ] = ( ucontrol - > value . integer . value [ 0 ] ? 0 : 32 )
| ( amp - > amp_vol [ index ] [ 0 ] & 31 ) ;
vol [ 1 ] = ( ucontrol - > value . integer . value [ 1 ] ? 0 : 32 )
| ( amp - > amp_vol [ index ] [ 1 ] & 31 ) ;
2005-04-17 02:20:36 +04:00
return awacs_amp_set_vol ( amp , index , vol [ 0 ] , vol [ 1 ] , 1 ) ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_info_tone_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 1 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 14 ;
return 0 ;
}
2008-04-16 21:39:27 +04:00
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_tone_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2008-08-08 19:12:47 +04:00
2005-04-17 02:20:36 +04:00
ucontrol - > value . integer . value [ 0 ] = amp - > amp_tone [ index ] ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_tone_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int index = kcontrol - > private_value ;
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2007-11-15 18:14:12 +03:00
unsigned int val ;
2008-08-08 19:12:47 +04:00
2007-11-15 18:14:12 +03:00
val = ucontrol - > value . integer . value [ 0 ] ;
if ( val > 14 )
return - EINVAL ;
if ( val ! = amp - > amp_tone [ index ] ) {
amp - > amp_tone [ index ] = val ;
2005-04-17 02:20:36 +04:00
awacs_amp_set_tone ( amp , amp - > amp_tone [ 0 ] , amp - > amp_tone [ 1 ] ) ;
return 1 ;
}
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_info_master_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 1 ;
uinfo - > value . integer . min = 0 ;
uinfo - > value . integer . max = 99 ;
return 0 ;
}
2008-04-16 21:39:27 +04:00
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_get_master_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
struct awacs_amp * amp = chip - > mixer_data ;
2008-08-08 19:12:47 +04:00
2005-04-17 02:20:36 +04:00
ucontrol - > value . integer . value [ 0 ] = amp - > amp_master ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_put_master_amp ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
struct awacs_amp * amp = chip - > mixer_data ;
2007-11-15 18:14:12 +03:00
unsigned int val ;
2008-08-08 19:12:47 +04:00
2007-11-15 18:14:12 +03:00
val = ucontrol - > value . integer . value [ 0 ] ;
if ( val > 99 )
return - EINVAL ;
if ( val ! = amp - > amp_master ) {
amp - > amp_master = val ;
2005-04-17 02:20:36 +04:00
awacs_amp_set_master ( amp , amp - > amp_master ) ;
return 1 ;
}
return 0 ;
}
# define AMP_CH_SPK 0
# define AMP_CH_HD 1
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_amp_vol [ ] __initdata = {
2005-04-17 02:20:36 +04:00
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " PC Speaker Playback Volume " ,
. info = snd_pmac_awacs_info_volume_amp ,
. get = snd_pmac_awacs_get_volume_amp ,
. put = snd_pmac_awacs_put_volume_amp ,
. private_value = AMP_CH_SPK ,
} ,
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Headphone Playback Volume " ,
. info = snd_pmac_awacs_info_volume_amp ,
. get = snd_pmac_awacs_get_volume_amp ,
. put = snd_pmac_awacs_put_volume_amp ,
. private_value = AMP_CH_HD ,
} ,
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Tone Control - Bass " ,
. info = snd_pmac_awacs_info_tone_amp ,
. get = snd_pmac_awacs_get_tone_amp ,
. put = snd_pmac_awacs_put_tone_amp ,
. private_value = 0 ,
} ,
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Tone Control - Treble " ,
. info = snd_pmac_awacs_info_tone_amp ,
. get = snd_pmac_awacs_get_tone_amp ,
. put = snd_pmac_awacs_put_tone_amp ,
. private_value = 1 ,
} ,
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Amp Master Playback Volume " ,
. info = snd_pmac_awacs_info_master_amp ,
. get = snd_pmac_awacs_get_master_amp ,
. put = snd_pmac_awacs_put_master_amp ,
} ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_amp_hp_sw __initdata = {
2005-04-17 02:20:36 +04:00
. iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " Headphone Playback Switch " ,
. info = snd_pmac_boolean_stereo_info ,
. get = snd_pmac_awacs_get_switch_amp ,
. put = snd_pmac_awacs_put_switch_amp ,
. private_value = AMP_CH_HD ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_amp_spk_sw __initdata = {
2005-04-17 02:20:36 +04:00
. iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
. name = " PC Speaker Playback Switch " ,
. info = snd_pmac_boolean_stereo_info ,
. get = snd_pmac_awacs_get_switch_amp ,
. put = snd_pmac_awacs_put_switch_amp ,
. private_value = AMP_CH_SPK ,
} ;
# endif /* PMAC_AMP_AVAIL */
/*
* mic boost for screamer
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_screamer_mic_boost_info ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_info * uinfo )
2005-04-17 02:20:36 +04:00
{
uinfo - > type = SNDRV_CTL_ELEM_TYPE_INTEGER ;
uinfo - > count = 1 ;
uinfo - > value . integer . min = 0 ;
2008-04-17 19:55:30 +04:00
uinfo - > value . integer . max = 3 ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_screamer_mic_boost_get ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2008-04-17 19:55:30 +04:00
int val = 0 ;
2005-04-17 02:20:36 +04:00
unsigned long flags ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
if ( chip - > awacs_reg [ 6 ] & MASK_MIC_BOOST )
2008-04-17 19:55:30 +04:00
val | = 2 ;
if ( chip - > awacs_reg [ 0 ] & MASK_GAINLINE )
val | = 1 ;
2005-04-17 02:20:36 +04:00
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
ucontrol - > value . integer . value [ 0 ] = val ;
return 0 ;
}
2005-11-17 17:09:46 +03:00
static int snd_pmac_screamer_mic_boost_put ( struct snd_kcontrol * kcontrol ,
struct snd_ctl_elem_value * ucontrol )
2005-04-17 02:20:36 +04:00
{
2005-11-17 17:09:46 +03:00
struct snd_pmac * chip = snd_kcontrol_chip ( kcontrol ) ;
2005-04-17 02:20:36 +04:00
int changed = 0 ;
int val0 , val6 ;
unsigned long flags ;
spin_lock_irqsave ( & chip - > reg_lock , flags ) ;
val0 = chip - > awacs_reg [ 0 ] & ~ MASK_GAINLINE ;
val6 = chip - > awacs_reg [ 6 ] & ~ MASK_MIC_BOOST ;
2008-04-17 19:55:30 +04:00
if ( ucontrol - > value . integer . value [ 0 ] & 1 )
2005-04-17 02:20:36 +04:00
val0 | = MASK_GAINLINE ;
2008-04-17 19:55:30 +04:00
if ( ucontrol - > value . integer . value [ 0 ] & 2 )
val6 | = MASK_MIC_BOOST ;
2005-04-17 02:20:36 +04:00
if ( val0 ! = chip - > awacs_reg [ 0 ] ) {
snd_pmac_awacs_write_reg ( chip , 0 , val0 ) ;
changed = 1 ;
}
if ( val6 ! = chip - > awacs_reg [ 6 ] ) {
snd_pmac_awacs_write_reg ( chip , 6 , val6 ) ;
changed = 1 ;
}
spin_unlock_irqrestore ( & chip - > reg_lock , flags ) ;
return changed ;
}
/*
* lists of mixer elements
*/
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_mixers [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " Master Capture Switch " , 1 , SHIFT_LOOPTHRU , 0 ) ,
2008-04-17 19:55:30 +04:00
AWACS_VOLUME ( " Master Capture Volume " , 0 , 4 , 0 ) ,
/* AWACS_SWITCH("Unknown Playback Switch", 6, SHIFT_PAROUT0, 0), */
} ;
static struct snd_kcontrol_new snd_pmac_screamer_mixers_beige [ ] __initdata = {
AWACS_VOLUME ( " Master Playback Volume " , 2 , 6 , 1 ) ,
AWACS_VOLUME ( " Play-through Playback Volume " , 5 , 6 , 1 ) ,
AWACS_SWITCH ( " Line Capture Switch " , 0 , SHIFT_MUX_MIC , 0 ) ,
AWACS_SWITCH ( " CD Capture Switch " , 0 , SHIFT_MUX_LINE , 0 ) ,
} ;
2009-01-20 23:01:17 +03:00
static struct snd_kcontrol_new snd_pmac_screamer_mixers_lo [ ] __initdata = {
2008-04-17 19:55:30 +04:00
AWACS_VOLUME ( " Line out Playback Volume " , 2 , 6 , 1 ) ,
2009-01-20 23:01:17 +03:00
} ;
static struct snd_kcontrol_new snd_pmac_screamer_mixers_imac [ ] __initdata = {
AWACS_VOLUME ( " Play-through Playback Volume " , 5 , 6 , 1 ) ,
2008-04-17 19:55:30 +04:00
AWACS_SWITCH ( " CD Capture Switch " , 0 , SHIFT_MUX_CD , 0 ) ,
} ;
2008-08-25 10:02:12 +04:00
static struct snd_kcontrol_new snd_pmac_screamer_mixers_g4agp [ ] __initdata = {
AWACS_VOLUME ( " Line out Playback Volume " , 2 , 6 , 1 ) ,
AWACS_VOLUME ( " Master Playback Volume " , 5 , 6 , 1 ) ,
AWACS_SWITCH ( " CD Capture Switch " , 0 , SHIFT_MUX_CD , 0 ) ,
AWACS_SWITCH ( " Line Capture Switch " , 0 , SHIFT_MUX_MIC , 0 ) ,
} ;
2008-04-17 19:55:30 +04:00
static struct snd_kcontrol_new snd_pmac_awacs_mixers_pmac7500 [ ] __initdata = {
AWACS_VOLUME ( " Line out Playback Volume " , 2 , 6 , 1 ) ,
AWACS_SWITCH ( " CD Capture Switch " , 0 , SHIFT_MUX_CD , 0 ) ,
AWACS_SWITCH ( " Line Capture Switch " , 0 , SHIFT_MUX_MIC , 0 ) ,
} ;
2009-01-20 23:01:17 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_mixers_pmac5500 [ ] __initdata = {
AWACS_VOLUME ( " Headphone Playback Volume " , 2 , 6 , 1 ) ,
} ;
2008-04-17 19:55:30 +04:00
static struct snd_kcontrol_new snd_pmac_awacs_mixers_pmac [ ] __initdata = {
AWACS_VOLUME ( " Master Playback Volume " , 2 , 6 , 1 ) ,
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " CD Capture Switch " , 0 , SHIFT_MUX_CD , 0 ) ,
} ;
/* FIXME: is this correct order?
* screamer ( powerbook G3 pismo ) seems to have different bits . . .
*/
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_mixers2 [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " Line Capture Switch " , 0 , SHIFT_MUX_LINE , 0 ) ,
AWACS_SWITCH ( " Mic Capture Switch " , 0 , SHIFT_MUX_MIC , 0 ) ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_screamer_mixers2 [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " Line Capture Switch " , 0 , SHIFT_MUX_MIC , 0 ) ,
AWACS_SWITCH ( " Mic Capture Switch " , 0 , SHIFT_MUX_LINE , 0 ) ,
} ;
2009-01-20 23:01:17 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_mixers2_pmac5500 [ ] __initdata = {
AWACS_SWITCH ( " CD Capture Switch " , 0 , SHIFT_MUX_CD , 0 ) ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_master_sw __initdata =
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " Master Playback Switch " , 1 , SHIFT_HDMUTE , 1 ) ;
2008-04-17 19:55:30 +04:00
static struct snd_kcontrol_new snd_pmac_awacs_master_sw_imac __initdata =
AWACS_SWITCH ( " Line out Playback Switch " , 1 , SHIFT_HDMUTE , 1 ) ;
2009-01-20 23:01:17 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_master_sw_pmac5500 __initdata =
AWACS_SWITCH ( " Headphone Playback Switch " , 1 , SHIFT_HDMUTE , 1 ) ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_mic_boost [ ] __initdata = {
2008-04-17 19:55:30 +04:00
AWACS_SWITCH ( " Mic Boost Capture Switch " , 0 , SHIFT_GAINLINE , 0 ) ,
2005-04-17 02:20:36 +04:00
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_screamer_mic_boost [ ] __initdata = {
2005-04-17 02:20:36 +04:00
{ . iface = SNDRV_CTL_ELEM_IFACE_MIXER ,
2008-04-17 19:55:30 +04:00
. name = " Mic Boost Capture Volume " ,
2005-04-17 02:20:36 +04:00
. info = snd_pmac_screamer_mic_boost_info ,
. get = snd_pmac_screamer_mic_boost_get ,
. put = snd_pmac_screamer_mic_boost_put ,
} ,
} ;
2008-04-17 19:55:30 +04:00
static struct snd_kcontrol_new snd_pmac_awacs_mic_boost_pmac7500 [ ] __initdata =
{
AWACS_SWITCH ( " Line Boost Capture Switch " , 0 , SHIFT_GAINLINE , 0 ) ,
} ;
static struct snd_kcontrol_new snd_pmac_screamer_mic_boost_beige [ ] __initdata =
{
AWACS_SWITCH ( " Line Boost Capture Switch " , 0 , SHIFT_GAINLINE , 0 ) ,
AWACS_SWITCH ( " CD Boost Capture Switch " , 6 , SHIFT_MIC_BOOST , 0 ) ,
} ;
static struct snd_kcontrol_new snd_pmac_screamer_mic_boost_imac [ ] __initdata =
{
AWACS_SWITCH ( " Line Boost Capture Switch " , 0 , SHIFT_GAINLINE , 0 ) ,
AWACS_SWITCH ( " Mic Boost Capture Switch " , 6 , SHIFT_MIC_BOOST , 0 ) ,
} ;
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_speaker_vol [ ] __initdata = {
2005-04-17 02:20:36 +04:00
AWACS_VOLUME ( " PC Speaker Playback Volume " , 4 , 6 , 1 ) ,
} ;
2008-04-17 19:55:30 +04:00
2005-11-17 17:09:46 +03:00
static struct snd_kcontrol_new snd_pmac_awacs_speaker_sw __initdata =
2005-04-17 02:20:36 +04:00
AWACS_SWITCH ( " PC Speaker Playback Switch " , 1 , SHIFT_SPKMUTE , 1 ) ;
2008-08-25 10:04:23 +04:00
static struct snd_kcontrol_new snd_pmac_awacs_speaker_sw_imac1 __initdata =
AWACS_SWITCH ( " PC Speaker Playback Switch " , 1 , SHIFT_PAROUT1 , 1 ) ;
static struct snd_kcontrol_new snd_pmac_awacs_speaker_sw_imac2 __initdata =
2008-04-17 19:55:30 +04:00
AWACS_SWITCH ( " PC Speaker Playback Switch " , 1 , SHIFT_PAROUT1 , 0 ) ;
2005-04-17 02:20:36 +04:00
/*
* add new mixer elements to the card
*/
2008-04-16 21:39:27 +04:00
static int build_mixers ( struct snd_pmac * chip , int nums ,
struct snd_kcontrol_new * mixers )
2005-04-17 02:20:36 +04:00
{
int i , err ;
for ( i = 0 ; i < nums ; i + + ) {
2008-04-16 21:39:27 +04:00
err = snd_ctl_add ( chip - > card , snd_ctl_new1 ( & mixers [ i ] , chip ) ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
}
return 0 ;
}
/*
* restore all registers
*/
2005-11-17 17:09:46 +03:00
static void awacs_restore_all_regs ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
snd_pmac_awacs_write_noreg ( chip , 0 , chip - > awacs_reg [ 0 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 1 , chip - > awacs_reg [ 1 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 2 , chip - > awacs_reg [ 2 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 4 , chip - > awacs_reg [ 4 ] ) ;
if ( chip - > model = = PMAC_SCREAMER ) {
snd_pmac_awacs_write_noreg ( chip , 5 , chip - > awacs_reg [ 5 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 6 , chip - > awacs_reg [ 6 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 7 , chip - > awacs_reg [ 7 ] ) ;
}
}
2005-06-28 01:36:34 +04:00
# ifdef CONFIG_PM
2005-11-17 17:09:46 +03:00
static void snd_pmac_awacs_suspend ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
snd_pmac_awacs_write_noreg ( chip , 1 , ( chip - > awacs_reg [ 1 ]
| MASK_AMUTE | MASK_CMUTE ) ) ;
}
2005-11-17 17:09:46 +03:00
static void snd_pmac_awacs_resume ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
if ( machine_is_compatible ( " PowerBook3,1 " )
| | machine_is_compatible ( " PowerBook3,2 " ) ) {
2005-07-09 12:53:24 +04:00
msleep ( 100 ) ;
2005-04-17 02:20:36 +04:00
snd_pmac_awacs_write_reg ( chip , 1 ,
chip - > awacs_reg [ 1 ] & ~ MASK_PAROUT ) ;
2005-07-09 12:53:24 +04:00
msleep ( 300 ) ;
2005-04-17 02:20:36 +04:00
}
awacs_restore_all_regs ( chip ) ;
if ( chip - > model = = PMAC_SCREAMER ) {
/* reset power bits in reg 6 */
mdelay ( 5 ) ;
snd_pmac_awacs_write_noreg ( chip , 6 , chip - > awacs_reg [ 6 ] ) ;
}
screamer_recalibrate ( chip ) ;
# ifdef PMAC_AMP_AVAIL
if ( chip - > mixer_data ) {
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2008-04-16 21:39:27 +04:00
awacs_amp_set_vol ( amp , 0 ,
amp - > amp_vol [ 0 ] [ 0 ] , amp - > amp_vol [ 0 ] [ 1 ] , 0 ) ;
awacs_amp_set_vol ( amp , 1 ,
amp - > amp_vol [ 1 ] [ 0 ] , amp - > amp_vol [ 1 ] [ 1 ] , 0 ) ;
2005-04-17 02:20:36 +04:00
awacs_amp_set_tone ( amp , amp - > amp_tone [ 0 ] , amp - > amp_tone [ 1 ] ) ;
awacs_amp_set_master ( amp , amp - > amp_master ) ;
}
# endif
}
2005-06-28 01:36:34 +04:00
# endif /* CONFIG_PM */
2005-04-17 02:20:36 +04:00
2009-01-20 23:01:15 +03:00
# define IS_PM7500 (machine_is_compatible("AAPL,7500") \
| | machine_is_compatible ( " AAPL,8500 " ) \
| | machine_is_compatible ( " AAPL,9500 " ) )
2009-01-20 23:01:13 +03:00
# define IS_PM5500 (machine_is_compatible("AAPL,e411"))
2008-04-17 19:55:30 +04:00
# define IS_BEIGE (machine_is_compatible("AAPL,Gossamer"))
2008-08-25 10:04:23 +04:00
# define IS_IMAC1 (machine_is_compatible("PowerMac2,1"))
# define IS_IMAC2 (machine_is_compatible("PowerMac2,2") \
2008-04-17 19:55:30 +04:00
| | machine_is_compatible ( " PowerMac4,1 " ) )
2008-08-25 10:02:12 +04:00
# define IS_G4AGP (machine_is_compatible("PowerMac3,1"))
2009-01-20 23:01:14 +03:00
# define IS_LOMBARD (machine_is_compatible("PowerBook1,1"))
2008-04-17 19:55:30 +04:00
2008-08-25 10:04:23 +04:00
static int imac1 , imac2 ;
2008-04-17 19:55:30 +04:00
2005-04-17 02:20:36 +04:00
# ifdef PMAC_SUPPORT_AUTOMUTE
/*
* auto - mute stuffs
*/
2005-11-17 17:09:46 +03:00
static int snd_pmac_awacs_detect_headphone ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
return ( in_le32 ( & chip - > awacs - > codec_stat ) & chip - > hp_stat_mask ) ? 1 : 0 ;
}
# ifdef PMAC_AMP_AVAIL
2005-11-17 17:09:46 +03:00
static int toggle_amp_mute ( struct awacs_amp * amp , int index , int mute )
2005-04-17 02:20:36 +04:00
{
int vol [ 2 ] ;
vol [ 0 ] = amp - > amp_vol [ index ] [ 0 ] & 31 ;
vol [ 1 ] = amp - > amp_vol [ index ] [ 1 ] & 31 ;
if ( mute ) {
vol [ 0 ] | = 32 ;
vol [ 1 ] | = 32 ;
}
return awacs_amp_set_vol ( amp , index , vol [ 0 ] , vol [ 1 ] , 1 ) ;
}
# endif
2005-11-17 17:09:46 +03:00
static void snd_pmac_awacs_update_automute ( struct snd_pmac * chip , int do_notify )
2005-04-17 02:20:36 +04:00
{
if ( chip - > auto_mute ) {
# ifdef PMAC_AMP_AVAIL
if ( chip - > mixer_data ) {
2005-11-17 17:09:46 +03:00
struct awacs_amp * amp = chip - > mixer_data ;
2005-04-17 02:20:36 +04:00
int changed ;
if ( snd_pmac_awacs_detect_headphone ( chip ) ) {
changed = toggle_amp_mute ( amp , AMP_CH_HD , 0 ) ;
changed | = toggle_amp_mute ( amp , AMP_CH_SPK , 1 ) ;
} else {
changed = toggle_amp_mute ( amp , AMP_CH_HD , 1 ) ;
changed | = toggle_amp_mute ( amp , AMP_CH_SPK , 0 ) ;
}
if ( do_notify & & ! changed )
return ;
} else
# endif
{
2008-04-17 19:55:30 +04:00
int reg = chip - > awacs_reg [ 1 ]
| ( MASK_HDMUTE | MASK_SPKMUTE ) ;
2008-08-25 10:04:23 +04:00
if ( imac1 ) {
reg & = ~ MASK_SPKMUTE ;
reg | = MASK_PAROUT1 ;
} else if ( imac2 ) {
2008-04-17 19:55:30 +04:00
reg & = ~ MASK_SPKMUTE ;
reg & = ~ MASK_PAROUT1 ;
}
2005-04-17 02:20:36 +04:00
if ( snd_pmac_awacs_detect_headphone ( chip ) )
reg & = ~ MASK_HDMUTE ;
2008-08-25 10:04:23 +04:00
else if ( imac1 )
reg & = ~ MASK_PAROUT1 ;
else if ( imac2 )
2008-04-17 19:55:30 +04:00
reg | = MASK_PAROUT1 ;
2005-04-17 02:20:36 +04:00
else
reg & = ~ MASK_SPKMUTE ;
if ( do_notify & & reg = = chip - > awacs_reg [ 1 ] )
return ;
snd_pmac_awacs_write_reg ( chip , 1 , reg ) ;
}
if ( do_notify ) {
snd_ctl_notify ( chip - > card , SNDRV_CTL_EVENT_MASK_VALUE ,
& chip - > master_sw_ctl - > id ) ;
snd_ctl_notify ( chip - > card , SNDRV_CTL_EVENT_MASK_VALUE ,
& chip - > speaker_sw_ctl - > id ) ;
snd_ctl_notify ( chip - > card , SNDRV_CTL_EVENT_MASK_VALUE ,
& chip - > hp_detect_ctl - > id ) ;
}
}
}
# endif /* PMAC_SUPPORT_AUTOMUTE */
/*
* initialize chip
*/
int __init
2005-11-17 17:09:46 +03:00
snd_pmac_awacs_init ( struct snd_pmac * chip )
2005-04-17 02:20:36 +04:00
{
2008-04-17 19:55:30 +04:00
int pm7500 = IS_PM7500 ;
2009-01-20 23:01:13 +03:00
int pm5500 = IS_PM5500 ;
2008-04-17 19:55:30 +04:00
int beige = IS_BEIGE ;
2008-08-25 10:02:12 +04:00
int g4agp = IS_G4AGP ;
2009-01-20 23:01:14 +03:00
int lombard = IS_LOMBARD ;
2008-08-25 10:04:23 +04:00
int imac ;
2005-04-17 02:20:36 +04:00
int err , vol ;
2009-01-20 23:01:17 +03:00
struct snd_kcontrol * vmaster_sw , * vmaster_vol ;
struct snd_kcontrol * master_vol , * speaker_vol ;
2005-04-17 02:20:36 +04:00
2008-08-25 10:04:23 +04:00
imac1 = IS_IMAC1 ;
imac2 = IS_IMAC2 ;
imac = imac1 | | imac2 ;
2005-04-17 02:20:36 +04:00
/* looks like MASK_GAINLINE triggers something, so we set here
* as start - up
*/
chip - > awacs_reg [ 0 ] = MASK_MUX_CD | 0xff | MASK_GAINLINE ;
chip - > awacs_reg [ 1 ] = MASK_CMUTE | MASK_AMUTE ;
/* FIXME: Only machines with external SRS module need MASK_PAROUT */
if ( chip - > has_iic | | chip - > device_id = = 0x5 | |
2008-04-16 21:39:27 +04:00
/* chip->_device_id == 0x8 || */
2005-04-17 02:20:36 +04:00
chip - > device_id = = 0xb )
chip - > awacs_reg [ 1 ] | = MASK_PAROUT ;
/* get default volume from nvram */
// vol = (~nvram_read_byte(0x1308) & 7) << 1;
// vol = ((pmac_xpram_read( 8 ) & 7 ) << 1 );
vol = 0x0f ; /* no, on alsa, muted as default */
vol = vol + ( vol < < 6 ) ;
chip - > awacs_reg [ 2 ] = vol ;
chip - > awacs_reg [ 4 ] = vol ;
if ( chip - > model = = PMAC_SCREAMER ) {
2008-04-16 21:39:27 +04:00
/* FIXME: screamer has loopthru vol control */
chip - > awacs_reg [ 5 ] = vol ;
/* FIXME: maybe should be vol << 3 for PCMCIA speaker */
chip - > awacs_reg [ 6 ] = MASK_MIC_BOOST ;
2005-04-17 02:20:36 +04:00
chip - > awacs_reg [ 7 ] = 0 ;
}
awacs_restore_all_regs ( chip ) ;
chip - > manufacturer = ( in_le32 ( & chip - > awacs - > codec_stat ) > > 8 ) & 0xf ;
screamer_recalibrate ( chip ) ;
chip - > revision = ( in_le32 ( & chip - > awacs - > codec_stat ) > > 12 ) & 0xf ;
# ifdef PMAC_AMP_AVAIL
if ( chip - > revision = = 3 & & chip - > has_iic & & CHECK_CUDA_AMP ( ) ) {
2006-07-25 17:28:03 +04:00
struct awacs_amp * amp = kzalloc ( sizeof ( * amp ) , GFP_KERNEL ) ;
2005-04-17 02:20:36 +04:00
if ( ! amp )
return - ENOMEM ;
chip - > mixer_data = amp ;
chip - > mixer_free = awacs_amp_free ;
2008-04-16 21:39:27 +04:00
/* mute and zero vol */
awacs_amp_set_vol ( amp , 0 , 63 , 63 , 0 ) ;
2005-04-17 02:20:36 +04:00
awacs_amp_set_vol ( amp , 1 , 63 , 63 , 0 ) ;
awacs_amp_set_tone ( amp , 7 , 7 ) ; /* 0 dB */
awacs_amp_set_master ( amp , 79 ) ; /* 0 dB */
}
# endif /* PMAC_AMP_AVAIL */
if ( chip - > hp_stat_mask = = 0 ) {
/* set headphone-jack detection bit */
switch ( chip - > model ) {
case PMAC_AWACS :
2009-01-20 23:01:13 +03:00
chip - > hp_stat_mask = pm7500 | | pm5500 ? MASK_HDPCONN
2008-04-17 19:55:30 +04:00
: MASK_LOCONN ;
2005-04-17 02:20:36 +04:00
break ;
case PMAC_SCREAMER :
switch ( chip - > device_id ) {
case 0x08 :
2008-04-17 19:55:30 +04:00
case 0x0B :
chip - > hp_stat_mask = imac
? MASK_LOCONN_IMAC |
MASK_HDPLCONN_IMAC |
MASK_HDPRCONN_IMAC
: MASK_HDPCONN ;
2005-04-17 02:20:36 +04:00
break ;
case 0x00 :
case 0x05 :
2008-04-17 19:55:30 +04:00
chip - > hp_stat_mask = MASK_LOCONN ;
2005-04-17 02:20:36 +04:00
break ;
default :
2008-04-17 19:55:30 +04:00
chip - > hp_stat_mask = MASK_HDPCONN ;
2005-04-17 02:20:36 +04:00
break ;
}
break ;
default :
snd_BUG ( ) ;
break ;
}
}
/*
* build mixers
*/
strcpy ( chip - > card - > mixername , " PowerMac AWACS " ) ;
2008-04-16 21:39:27 +04:00
err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_mixers ) ,
snd_pmac_awacs_mixers ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
2008-08-25 10:02:12 +04:00
if ( beige | | g4agp )
2008-04-17 19:55:30 +04:00
;
2009-01-20 23:01:13 +03:00
else if ( chip - > model = = PMAC_SCREAMER | | pm5500 )
2005-04-17 02:20:36 +04:00
err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_screamer_mixers2 ) ,
snd_pmac_screamer_mixers2 ) ;
2008-04-17 19:55:30 +04:00
else if ( ! pm7500 )
2005-04-17 02:20:36 +04:00
err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_mixers2 ) ,
snd_pmac_awacs_mixers2 ) ;
if ( err < 0 )
return err ;
2009-01-20 23:01:17 +03:00
if ( pm5500 ) {
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_awacs_mixers2_pmac5500 ) ,
snd_pmac_awacs_mixers2_pmac5500 ) ;
if ( err < 0 )
return err ;
}
2008-04-17 19:55:30 +04:00
if ( pm7500 )
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_awacs_mixers_pmac7500 ) ,
snd_pmac_awacs_mixers_pmac7500 ) ;
2009-01-20 23:01:17 +03:00
else if ( pm5500 )
err = snd_ctl_add ( chip - > card ,
( master_vol = snd_ctl_new1 ( snd_pmac_awacs_mixers_pmac5500 ,
chip ) ) ) ;
2008-04-17 19:55:30 +04:00
else if ( beige )
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_screamer_mixers_beige ) ,
snd_pmac_screamer_mixers_beige ) ;
2009-01-20 23:01:17 +03:00
else if ( imac | | lombard ) {
err = snd_ctl_add ( chip - > card ,
( master_vol = snd_ctl_new1 ( snd_pmac_screamer_mixers_lo ,
chip ) ) ) ;
if ( err < 0 )
return err ;
2008-04-17 19:55:30 +04:00
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_screamer_mixers_imac ) ,
snd_pmac_screamer_mixers_imac ) ;
2009-01-20 23:01:17 +03:00
} else if ( g4agp )
2008-08-25 10:02:12 +04:00
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_screamer_mixers_g4agp ) ,
snd_pmac_screamer_mixers_g4agp ) ;
2008-04-17 19:55:30 +04:00
else
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_awacs_mixers_pmac ) ,
snd_pmac_awacs_mixers_pmac ) ;
if ( err < 0 )
return err ;
2009-01-20 23:01:14 +03:00
chip - > master_sw_ctl = snd_ctl_new1 ( ( pm7500 | | imac | | g4agp | | lombard )
2008-04-17 19:55:30 +04:00
? & snd_pmac_awacs_master_sw_imac
2009-01-20 23:01:17 +03:00
: pm5500
? & snd_pmac_awacs_master_sw_pmac5500
2008-04-17 19:55:30 +04:00
: & snd_pmac_awacs_master_sw , chip ) ;
2008-04-16 21:39:27 +04:00
err = snd_ctl_add ( chip - > card , chip - > master_sw_ctl ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
# ifdef PMAC_AMP_AVAIL
if ( chip - > mixer_data ) {
/* use amplifier. the signal is connected from route A
* to the amp . the amp has its headphone and speaker
* volumes and mute switches , so we use them instead of
* screamer registers .
* in this case , it seems the route C is not used .
*/
2008-04-16 21:39:27 +04:00
err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_amp_vol ) ,
snd_pmac_awacs_amp_vol ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
/* overwrite */
2008-04-16 21:39:27 +04:00
chip - > master_sw_ctl = snd_ctl_new1 ( & snd_pmac_awacs_amp_hp_sw ,
chip ) ;
err = snd_ctl_add ( chip - > card , chip - > master_sw_ctl ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
2008-04-16 21:39:27 +04:00
chip - > speaker_sw_ctl = snd_ctl_new1 ( & snd_pmac_awacs_amp_spk_sw ,
chip ) ;
err = snd_ctl_add ( chip - > card , chip - > speaker_sw_ctl ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
} else
# endif /* PMAC_AMP_AVAIL */
{
/* route A = headphone, route C = speaker */
2009-01-20 23:01:17 +03:00
err = snd_ctl_add ( chip - > card ,
( speaker_vol = snd_ctl_new1 ( snd_pmac_awacs_speaker_vol ,
chip ) ) ) ;
2008-04-16 21:39:27 +04:00
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
2008-08-25 10:04:23 +04:00
chip - > speaker_sw_ctl = snd_ctl_new1 ( imac1
? & snd_pmac_awacs_speaker_sw_imac1
: imac2
? & snd_pmac_awacs_speaker_sw_imac2
2008-04-17 19:55:30 +04:00
: & snd_pmac_awacs_speaker_sw , chip ) ;
2008-04-16 21:39:27 +04:00
err = snd_ctl_add ( chip - > card , chip - > speaker_sw_ctl ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
}
2009-01-20 23:01:17 +03:00
if ( pm5500 | | imac | | lombard ) {
vmaster_sw = snd_ctl_make_virtual_master (
" Master Playback Switch " , ( unsigned int * ) NULL ) ;
err = snd_ctl_add_slave_uncached ( vmaster_sw ,
chip - > master_sw_ctl ) ;
if ( err < 0 )
return err ;
err = snd_ctl_add_slave_uncached ( vmaster_sw ,
chip - > speaker_sw_ctl ) ;
if ( err < 0 )
return err ;
err = snd_ctl_add ( chip - > card , vmaster_sw ) ;
if ( err < 0 )
return err ;
vmaster_vol = snd_ctl_make_virtual_master (
" Master Playback Volume " , ( unsigned int * ) NULL ) ;
err = snd_ctl_add_slave ( vmaster_vol , master_vol ) ;
if ( err < 0 )
return err ;
err = snd_ctl_add_slave ( vmaster_vol , speaker_vol ) ;
if ( err < 0 )
return err ;
err = snd_ctl_add ( chip - > card , vmaster_vol ) ;
if ( err < 0 )
return err ;
}
2008-08-25 10:02:12 +04:00
if ( beige | | g4agp )
2008-04-17 19:55:30 +04:00
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_screamer_mic_boost_beige ) ,
snd_pmac_screamer_mic_boost_beige ) ;
else if ( imac )
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_screamer_mic_boost_imac ) ,
snd_pmac_screamer_mic_boost_imac ) ;
else if ( chip - > model = = PMAC_SCREAMER )
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_screamer_mic_boost ) ,
snd_pmac_screamer_mic_boost ) ;
else if ( pm7500 )
err = build_mixers ( chip ,
ARRAY_SIZE ( snd_pmac_awacs_mic_boost_pmac7500 ) ,
snd_pmac_awacs_mic_boost_pmac7500 ) ;
else
err = build_mixers ( chip , ARRAY_SIZE ( snd_pmac_awacs_mic_boost ) ,
snd_pmac_awacs_mic_boost ) ;
if ( err < 0 )
return err ;
2005-04-17 02:20:36 +04:00
/*
* set lowlevel callbacks
*/
chip - > set_format = snd_pmac_awacs_set_format ;
2005-06-28 01:36:34 +04:00
# ifdef CONFIG_PM
2005-04-17 02:20:36 +04:00
chip - > suspend = snd_pmac_awacs_suspend ;
chip - > resume = snd_pmac_awacs_resume ;
# endif
# ifdef PMAC_SUPPORT_AUTOMUTE
2008-04-16 21:39:27 +04:00
err = snd_pmac_add_automute ( chip ) ;
if ( err < 0 )
2005-04-17 02:20:36 +04:00
return err ;
chip - > detect_headphone = snd_pmac_awacs_detect_headphone ;
chip - > update_automute = snd_pmac_awacs_update_automute ;
snd_pmac_awacs_update_automute ( chip , 0 ) ; /* update the status only */
# endif
if ( chip - > model = = PMAC_SCREAMER ) {
snd_pmac_awacs_write_noreg ( chip , 6 , chip - > awacs_reg [ 6 ] ) ;
snd_pmac_awacs_write_noreg ( chip , 0 , chip - > awacs_reg [ 0 ] ) ;
}
return 0 ;
}